Назад к вопросам
МВидео
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
}