Sobes.tech
Назад к вопросам
Middle+
4

Задача: «Параллельная обработка URL с ограничением времени» Напишите функцию fetchAll(urls []string, maxParallel int, timeout time.Duration) map[string]string, которая: 1. Для каждого URL выполняет HTTP GET-запрос (можно http.Get). 2. Возвращает map[string]string, где ключ — URL, значение — тело ответа (как строка) или пустая строка при ошибке/timауте. 3. Одновременно выполняется не более maxParallel запросов. 4. Общее время работы функции не должно превышать timeout. Если таймаут истёк, все незавершённые запросы прерываются, и функция возвращает то, что успело. 5. Используйте контекст для отмены.

Компании, где спрашивали
МВидеоМВидео

Ответ от нейросети

sobes.tech AI

go package main

import ( "context" "io/ioutil" "net/http" "sync" "time" )

func fetchAll(urls []string, maxParallel int, timeout time.Duration) map[string]string { ctx, cancel := context.WithTimeout(context.Background(), timeout) defer cancel()

results := make(map[string]string)
var mu sync.Mutex
sem := make(chan struct{}, maxParallel) // семафор для ограничения параллелизма
var wg sync.WaitGroup

for _, url := range urls {
	select {
	case <-ctx.Done():
		break
	default:
		wg.Add(1)
		sem <- struct{}{} // захватываем слот

		go func(u string) {
			defer wg.Done()
			defer func() { <-sem }() // освобождаем слот

			req, err := http.NewRequestWithContext(ctx, "GET", u, nil)
			if err != nil {
				mu.Lock()
				results[u] = ""
				mu.Unlock()
				return
			}

			resp, err := http.DefaultClient.Do(req)
			if err != nil {
				mu.Lock()
				results[u] = ""
				mu.Unlock()
				return
			}
			defer resp.Body.Close()

			body, err := ioutil.ReadAll(resp.Body)
			if err != nil {
				mu.Lock()
				results[u] = ""
				mu.Unlock()
				return
			}

			mu.Lock()
			results[u] = string(body)
			mu.Unlock()
		}(url)
	}
}

wg.Wait()
return results

}