Mono BLOG

Blog about Microsoft technologies (.NET, ASP.NET Core, Blazor, EF Core, WPF, TypeScript, etc.)

Task in C#

in .NET

The Task class in C# is a powerful tool for multithreaded programming, enabling asynchronous operations, parallel processing, and task cancellation. This guide demonstrates common patterns and best practices for leveraging Task effectively.


1. Executing Asynchronous Operations

Use Task.Run to offload work to the thread pool and await to handle completion.

static async Task<int> LongRunningOperationAsync()
{
    await Task.Delay(1000);  // Simulates async work (e.g., I/O)
    return 1;
}

static async Task MyMethodAsync()
{
    int result = await Task.Run(() => LongRunningOperationAsync());
    Console.WriteLine(result);  // Output: 1
}

Key Points: • Task.Run queues work to the thread pool. • await asynchronously waits for completion without blocking the thread.


2. Parallel Task Processing

Use Task.WhenAll to await multiple tasks concurrently.

Task<int> task1 = Task.Run(() => LongRunningOperationAsync());
Task<int> task2 = Task.Run(() => LongRunningOperationAsync());
Task<int> task3 = Task.Run(() => LongRunningOperationAsync());

int[] results = await Task.WhenAll(task1, task2, task3);
Console.WriteLine($"Sum: {results.Sum()}");  // Output: Sum: 3

Advantages: • Runs tasks in parallel for better CPU utilization. • Aggregates results when all tasks complete.


3. Canceling Async Operations

Use CancellationTokenSource to gracefully terminate tasks.

static async Task<int> LongRunningOperationAsync(CancellationToken cancellationToken)
{
    await Task.Delay(1000, cancellationToken);  // Honors cancellation
    return 1;
}

static async Task MyMethodAsync(CancellationToken cancellationToken)
{
    using var cts = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken);
    cts.CancelAfter(500);  // Auto-cancel after 500ms
    
    try
    {
        int result = await Task.Run(() => LongRunningOperationAsync(cts.Token));
        Console.WriteLine(result);
    }
    catch (TaskCanceledException)
    {
        Console.WriteLine("Operation canceled.");
    }
}

Mechanism: • CancellationToken propagates cancellation requests. • CancelAfter() triggers cancellation after a timeout.


Best Practices and Pitfalls

✅ Do:

• Use ConfigureAwait(false) in library code to avoid deadlocks. • Prefer async/await over .Result or .Wait() to prevent blocking. • Handle exceptions with try/catch around awaited tasks.

❌ Avoid:

• Mixing synchronous and async code without proper context management. • Ignoring OperationCanceledException when using cancellation tokens. • Overusing Task.Run for CPU-bound work; consider Parallel.For instead.


Summary

The Task API enables: • Asynchronous Programming: Non-blocking execution for I/O-bound work. • Parallelism: Efficient utilization of multi-core CPUs. • Cancellation: Graceful termination of long-running operations.

By following these patterns, you’ll write responsive, scalable, and maintainable C# applications. For advanced scenarios, explore ValueTask, TaskCompletionSource, and the TPL Dataflow library.

graph TD
    A[Start Task] --> B{Is Async?}
    B -->|Yes| C[Use async/await]
    B -->|No| D[Use Task.Run]
    C --> E[Handle Completion/Errors]
    D --> E
    E --> F[Process Results]