Task Parallel Library (TPL) in C#

Understanding Task Parallel Library (TPL) in C#

The **Task Parallel Library (TPL)** in C# provides a **higher-level abstraction** for managing parallel execution. It is more efficient and flexible than manually creating **threads**, allowing tasks to be scheduled and executed efficiently using the **Task class**.

Key Features of Task Parallel Library

  • Automatically manages **thread pooling**.
  • Supports **cancellation and continuation**.
  • Scales efficiently using **multi-core processors**.
  • Provides **async and await** integration for non-blocking execution.

Creating and Running Tasks

The **Task.Run()** method is used to create and execute a task asynchronously.

Example: Creating and Running a Task

using System;
using System.Threading.Tasks;

class Program
{
    static void Main()
    {
        Task task = Task.Run(() =>
        {
            Console.WriteLine($"Task {Task.CurrentId} is running.");
        });

        task.Wait(); // Wait for task to complete
        Console.WriteLine("Task completed.");
    }
}

// Output:
// Task 1 is running.
// Task completed.
        

The **Task.Run()** method schedules a task to run asynchronously on a **thread pool thread**.

Running Multiple Tasks in Parallel

The **Task.WhenAll()** method waits for multiple tasks to complete before proceeding.

Example: Running Multiple Tasks Using Task.WhenAll()

using System;
using System.Threading.Tasks;

class Program
{
    static async Task Main()
    {
        Task task1 = Task.Run(() => Console.WriteLine("Task 1 running"));
        Task task2 = Task.Run(() => Console.WriteLine("Task 2 running"));

        await Task.WhenAll(task1, task2);
        Console.WriteLine("All tasks completed.");
    }
}

// Output:
// Task 1 running
// Task 2 running
// All tasks completed.
        

**Task.WhenAll()** ensures that all tasks complete before executing the next statement.

Using Task Continuations

The **ContinueWith()** method allows chaining tasks, ensuring that a follow-up task executes after the first one completes.

Example: Task Continuation Using ContinueWith()

using System;
using System.Threading.Tasks;

class Program
{
    static void Main()
    {
        Task task = Task.Run(() => Console.WriteLine("Task 1 running"))
                        .ContinueWith(t => Console.WriteLine("Task 2 running"));

        task.Wait();
    }
}

// Output:
// Task 1 running
// Task 2 running
        

The **ContinueWith()** method ensures that **Task 2 runs only after Task 1 completes**.

Cancelling a Running Task

You can use a **CancellationToken** to cancel a running task.

Example: Cancelling a Task Using CancellationToken

using System;
using System.Threading;
using System.Threading.Tasks;

class Program
{
    static void Main()
    {
        CancellationTokenSource cts = new CancellationTokenSource();

        Task task = Task.Run(() =>
        {
            for (int i = 1; i <= 5; i++)
            {
                if (cts.Token.IsCancellationRequested)
                {
                    Console.WriteLine("Task cancelled.");
                    return;
                }
                Console.WriteLine($"Task running {i}");
                Thread.Sleep(1000);
            }
        }, cts.Token);

        Thread.Sleep(2000);
        cts.Cancel(); // Cancel the task
        task.Wait();
    }
}

// Output:
// Task running 1
// Task running 2
// Task cancelled.
        

The **CancellationTokenSource** allows gracefully stopping a task before it completes execution.

Best Practices for Using Task Parallel Library

  • Use **Task.Run()** for CPU-bound operations, avoiding manual thread creation.
  • Use **Task.WhenAll()** to efficiently execute multiple tasks concurrently.
  • Use **CancellationTokenSource** to stop long-running tasks gracefully.
  • Avoid using **Task.Wait() or Task.Result** in UI applications, as it can block the UI.