Processing Tasks as They Complete [ Documentation]

Task.WhenAny allows for starting multiple tasks at the same time and processing them one by one in the order in which they complete instead of the order in which they are started.

static async Task SumPageSizesAsync()
{
    var stopwatch = Stopwatch.StartNew();

    IEnumerable<Task<int>> downloadTasksQuery = from url in s_urlList
                                                select ProcessUrlAsync(url, s_client);

    // Start each Task since LINQ uses deferred execution otherwise:
    List<Task<int>> downloadTasks = downloadTasksQuery.ToList();

    int total = 0;

    while (downloadTasks.Any())
    {
        // Hold the next Task that finishes:
        Task<int> finishedTask = await Task.WhenAny(downloadTasks);

        downloadTasks.Remove(finishedTask); // Remove that Task from the list
        
        total += await finishedTask; // Unwrap the int from the Task, or, if faulted, throw the first child Exception
    }

    stopwatch.Stop();

    Console.WriteLine($"\nTotal bytes returned: {total:#,#}");
    Console.WriteLine($"Elapsed time: {stopwatch.Elapsed}\n");
}

see also

https://devblogs.microsoft.com/pfxteam/processing-tasks-as-they-complete/