Abstract [ Documentation] [ Documentation]

Asynchronous programming allows for the execution of long-running operations that do not block the main thread.

Concepts:

  • Synchronous operations complete sequentially; the next task does not start until the previous finishes.
  • Asynchronous operations allow for starting a task then, while waiting for that task to finish, starting another.
  • Parallel operations use multiple threads to execute several asynchronous tasks. They are deterministic: the order in which the tasks will complete is known.
  • Concurrent operations use multiple threads to execute several asynchronous tasks. They are nondeterministic: the order in which the tasks will complete is not known.

.NET’s asynchronous programming pattern described here is an implementation of the promise model of asynchrony which uses the futures and promises constructs.

The asynchronous programming models uses Task and Task<T> objects which model asynchronous operations. Generally:

  • For I/O-bound code, await a Task or Task<T> in a method marked async.
  • For CPU-bound code, await an operation that is started on a background thread via Task.Run.

An example of I/O-bound code: downloading data from a web service:

private readonly HttpClient _httpClient = new HttpClient();

downloadButton.Clicked += async (o, e) =>
{
    // This line will yield control to the UI as the request from the web service is happening.
    //
    // The UI thread is now free to perform other work.
    var stringData = await _httpClient.GetStringAsync(URL);
    DoSomethingWithData(stringData);
};

An example of CPU-bound code: performing a calculation in a game:

private DamageResult CalculateDamageDone()
{
    // Code omitted: Performs an expensive calculation and returns the result of that calculation.
}

calculateButton.Clicked += async (o, e) =>
{
    // This line will yield control to the UI while CalculateDamageDone() performs its work. The UI thread is free to perform other work.
    var damageResult = await Task.Run(() => CalculateDamageDone());
    DisplayDamage(damageResult);
};

Asynchronous vs. Synchronous

Asynchronous code like FryEggsAsync(2) does not block:

// the async modifier signals that the method contains an await
static async Task Main() 
{ 
	Egg eggs = await FryEggsAsync(2);
	Console.WriteLine("eggs are ready");
}

Synchronous code will block the thread executing it from doing any other work. It cannot be interrupted:

static void Main() 
{
	Egg eggs = FryEggs(2);
	Console.WriteLine("eggs are ready");
}

concurrent

Concurrent code like Task<Egg> eggsTask = FryEggsAsync(2) does not block and allows for concurrent execution:

static async Task Main() 
{
	Task<Egg> eggsTask = FryEggsAsync(2);
	Eggs eggs = await eggsTask;
	Console.WriteLine("eggs are ready");
}

composition

If any portion of an operation is asynchronous, the entire operation is asynchronous.
The composition of an asynchronous operation followed by synchronous work is asynchronous:

static async Task<Toast> MakeToastWithButterAndJamAsync() 
{
	var toast = await ToastBreadAsync();	// the asynchronous operation
	ApplyButter(toast);				// the synchronous operations
	ApplyJam(toast);
	
	return toast;
}

async

Async Code & LINQ

Use caution. LINQ uses deferred (lazy) execution, so async calls won’t happen immediately (unless iteration is forced via .ToList() or .ToArray())

Expression-Bodied Methods

If a method is expression-bodied, it can omit the async and await keywords:

static Task Main() => DoSomethingAsync();

…is equivalent to:

static async Task Main() 
{
	await DoSomethingAsync();
}

Other .NET Asynchronous Programming Patterns

These notes describe the Task Asynchronous Pattern, which is covered in more detail in the notes linked below.

ModelAbbrLanguage
constructs
Notes
Task Asynchronous Pattern
Modern, easier to use
TAPasync/awaitNotes on TAP
Asynchronous Programming Model
Legacy, more difficult to use
APMIAsyncResult, Begin, EndNotes on APM
Legacy pattern; not recommended for new development
Event-based Asynchronous PatternEAPEvents, event-handler delegates, EventArg-derived typesLegacy pattern; not recommended for new development

.NET also has a parallel programming pattern: the Task Parallel Library ( Notes on TPL).