I thought I understand the async-await pattern in C# but today I've found out I really do not.
In a simple code snippet like this. I have System.Net.ServicePointManager.DefaultConnectionLimit = 1000; defined already.
public static async Task Test()
{
    HttpClient client = new HttpClient();
    string str;
    for (int i = 1000; i <= 1100; i++)
        str = await client.GetStringAsync($"https://en.wikipedia.org/wiki/{i1}");
}
What does await do here? Initially I thought since this is in async-await pattern, it means basically HttpClient will initiate all HTTP GET calls in a multi-threaded fashion, i.e. basically all the Urls should be fetched at once.
But when I'm using Fiddler to analyze the behavior I've found it really fetches the URLs sequentially.
I need to change it to this to make it work:
public static async Task<int> Test()
{
    int ret = 0;
    HttpClient client = new HttpClient();
    List<Task> taskList = new List<Task>();
    for (int i = 1000; i <= 1100; i++)
    {
        var i1 = i;
        var task = Task.Run(() => client.GetStringAsync($"https://en.wikipedia.org/wiki/{i1}"));
        taskList.Add(task);
    }
    Task.WaitAll(taskList.ToArray());
    return ret;
}
This time the URLs are fetched in parallel. So what does the await keyword really do in the first code snippet?
The syntax async def introduces either a native coroutine or an asynchronous generator. The expressions async with and async for are also valid, and you'll see them later on. The keyword await passes function control back to the event loop. (It suspends the execution of the surrounding coroutine.)
await releases the current thread, but NOT to the thread pool. The UI thread doesn't come from the thread pool. If you run asynchronous method, e.g. ExecuteScalarAsync without async, await keywords, then this method will run asynchronously no matter what. The calling thread won't be affected .
Asynchronous functions are prefixed with the async keyword; await suspends the execution until an asynchronous function return promise is fulfilled and unwraps the value from the Promise returned.
Inside an async function, you can use the await keyword before a call to a function that returns a promise. This makes the code wait at that point until the promise is settled, at which point the fulfilled value of the promise is treated as a return value, or the rejected value is thrown.
It awaits the completion of the HTTP request. The code resumes (for iteration...) only after every single request is complete.
Your 2nd version works precisely because it doesn't await for each task to complete before initiating the following tasks, and only waits for all the tasks to complete after all have been started.
What async-await is useful for is allowing the calling function to continue doing other things while the asynchronous function is awaiting, as opposed to synchronous ("normal") functions that block the calling function until completion.
An await is an asynchronous wait. It is not a blocking call and allows the caller of your method to continue. The remainder of the code inside the method after an await will be executed when the Task returned has completed. 
In the first version of your code, you allow callers to continue. However, each iteration of the loop will wait until the Task returned by GetStringAsync has completed. This has the effect of sequentially downloading each URL, rather than concurrently.
Note that the second version of your code is not asynchronous insofar as it uses threads to perform the work in parallel.
If it were asynchronous, it would retrieve the webpage content using only one thread but still concurrently.
Something like this (untested):
public static async Task<int> Test()
{
    int ret = 0;
    HttpClient client = new HttpClient();
    List<Task> taskList = new List<Task>();
    for (int i = 1000; i <= 1100; i++)
    {
        var i1 = i;
        taskList.Add(client.GetStringAsync($"https://en.wikipedia.org/wiki/{i1}"));
    }
    await Task.WhenAll(taskList.ToArray());
    return ret;
}
Here, we start the tasks asynchronously and add them to the taskList. These tasks are non-blocking and will complete when the download has finished and the string retrieved. Pay attention to the call to Task.WhenAll rather than Task.WaitAll: the former is asynchronous and non-blocking, the latter is synchronous and blocking. This means that, at the await, the caller of this Test() method will receive the Task<int> returned: but the task will be incomplete until all of the strings are downloaded.
This is what forces async/await to proliferate throughout the stack. Once the very bottom call is asynchronous, it only makes sense if the rest of the callers all the way up are also asynchronous. Otherwise, you are forced to create a thread via Task.Run() calls or somesuch.
Per the msdn documentation
The await operator is applied to a task in an asynchronous method to suspend the execution of the method until the awaited task completes. The task represents ongoing work.
That means the await operator blocks the execution of the for loop until it get a responds from the server, making it sequential.
What you can do is create all the task (so that it begins execution) and then await all of them.
Here's an example from another StackOverflow question
public IEnumerable<TContent> DownloadContentFromUrls<TContent>(IEnumerable<string> urls)
{
    var queue = new ConcurrentQueue<TContent>();
    using (var client = new HttpClient())
    {
        Task.WaitAll(urls.Select(url =>
        {
            return client.GetAsync(url).ContinueWith(response =>
            {
                var content = JsonConvert.
                    DeserializeObject<IEnumerable<TContent>>(
                        response.Result.Content.ReadAsStringAsync().Result);
                foreach (var c in content)
                    queue.Enqueue(c);
            });
        }).ToArray());
    }
    return queue;
}
There's also good article in msdn that explains how to make parallel request with await.
Edit:
As @GaryMcLeanHall pointed out in a comment, you can change Task.WaitAll to await Task.WhenAll and add the async modifier to make the method return asynchronously
Here's another msdn article that picks the example in the first one and adds the use of WhenAll.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With