3 The async and await keywords

 

This chapter covers

  • Using Task and Task<T> to check if an operation has completed
  • Using Task and Task<T> to notify your code when the operation has completed
  • Using Task and Task<T> in synchronous code
  • How async-await works

In the previous chapter we looked at how the compiler can transform our code in order to add language features. In this chapter we’ll see how it applies to async-await.

Async-await is a feature that lets us write asynchronous code as if it was normal synchronous code. With asynchronous programming, when we perform an operation that would normally make the CPU wait, usually for data to arrive from some device (for example, reading a file), instead of waiting, we just do something else. Making asynchronous code look like normal code is kind of a big deal because traditionally you needed to divide each sequence of operations into little parts (breaking at each asynchronous operation) and call the right part at the part time. Unsurprisingly, this makes the code confusing to write.

3.1 The problem with asynchronous code

To demonstrate this, I want to place the first and last diagrams from chapter 1 side by side:

Figure 3.1 Logical flow vs. running asynchronously

It’s easy to see that the left side that is describes the logical flow is simple, linear and easy to understand, while the right side that describes how the asynchronous version is running is none of those things (it’s also very difficult to debug).

3.2 Introducing Task and Task<T>

3.2.1 Are we there yet?

3.2.2 Wake me up when we get there

3.2.3 The synchronous option

3.2.4 After the task has completed

3.3 How does async-await work?

3.4 async void methods

3.5 ValueTask and ValueTask<T>

3.6 And what about multithreading?

3.7 Summary

sitemap