5
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

C#Advent Calendar 2024

Day 12

【C#】非同期プログラミングの正しい理解と実践

Posted at

はじめに

非同期プログラミングは使いこなせていますか?

async/awaitは便利な機能ですが、正しく使わないとデッドロックやパフォーマンス低下などの問題を引き起こす可能性があります。

今回は、C#での非同期プログラミングについて、基礎から実践的な使い方をまとめてみました。

想定する読者

  • C#でのプログラミング経験がある初級者
  • 非同期プログラミングの概念を理解したい人
  • async/awaitの使い方に不安がある人

目次

  1. 非同期プログラミングとは
  2. 基本的な使い方
  3. よくあるミスと解決方法
  4. 実践的なシナリオ
  5. パフォーマンス最適化
  6. テスト方法

非同期プログラミングとは

モダンなアプリケーション開発において、非同期プログラミングは必須のスキルとなっています。ユーザー体験を損なうことなく、時間のかかる処理を実行するための大切な概念を説明していきます。

なぜ非同期処理が必要か?

アプリケーションの処理には、次のような時間のかかる操作が多く存在します。

  • ファイルの読み書き
  • データベースへのアクセス
  • Web APIの呼び出し
  • 重い計算処理

これらの処理を同期的に行うと、処理が完了するまでアプリケーションがフリーズしてしまい、ユーザー体験が著しく低下します。

同期処理と非同期処理の違い

【同期処理の場合】

public string ReadFileSync(string path)
{
    // この処理が完了するまで、次の処理に進めない
    return File.ReadAllText(path);
}

【非同期処理の場合】

public async Task<string> ReadFileAsync(string path)
{
    // 処理を待っている間に他の処理を実行できる
    return await File.ReadAllTextAsync(path);
}

基本的な使い方

async/awaitを使用した非同期プログラミングには、いくつかの基本的なパターンがあります。ここでは、よく使用される基本的な実装方法と、その背景にある考え方を解説します。

Taskクラスについて

Taskクラスは非同期操作を表現するクラスです。
Task<T>は非同期操作の結果として値を返す場合に使用します。

// 値を返さない非同期メソッド
public async Task DoWorkAsync()
{
    await Task.Delay(1000); // 1秒待機
    Console.WriteLine("作業完了");
}

// string型の値を返す非同期メソッド
public async Task<string> GetDataAsync()
{
    await Task.Delay(1000);
    return "データ";
}

async/awaitの基本

非同期メソッドを定義するにはasyncキーワードを使用し、非同期処理の待機にはawaitキーワードを使用します。

public async Task ProcessDataAsync()
{
    Console.WriteLine("処理開始");
    
    // GetDataAsyncの完了を待機
    string data = await GetDataAsync();
    
    Console.WriteLine($"取得したデータ: {data}");
    Console.WriteLine("処理完了");
}

よくあるミスと解決方法

非同期プログラミングには、経験者でも陥りやすいいくつかの落とし穴があります。

一般的なミスとその防ぎ方、そして適切な実装方法を整理しました。

1. async voidの使用

async voidは例外処理が難しく、テストも困難になるため、避けるべきです。

// 悪い例
public async void BadExample()
{
    await Task.Delay(100);
    throw new Exception("エラー発生"); // このエラーはキャッチできない
}

// 良い例
public async Task GoodExample()
{
    await Task.Delay(100);
    throw new Exception("エラー発生"); // このエラーは適切にハンドリングできる
}

2. awaitし忘れ

awaitを忘れると、非同期処理の完了を待たずに次の処理に進んでしまいます。

// 悪い例
public async Task BadExample()
{
    Task.Delay(1000); // await忘れ
    Console.WriteLine("完了"); // すぐに実行される
}

// 良い例
public async Task GoodExample()
{
    await Task.Delay(1000);
    Console.WriteLine("完了"); // 1秒後に実行される
}

実践的なシナリオ

実際の開発現場では、様々な状況で非同期プログラミングが必要となります。

ここでは、実際のプロジェクトでよく遭遇するシナリオとその実装方法を紹介します。

Web APIの呼び出し

実際のアプリケーションでよく使用するWeb APIの呼び出し例です。

public class WeatherService
{
    private readonly HttpClient _client;
    
    public WeatherService()
    {
        _client = new HttpClient();
    }

    public async Task<WeatherData> GetWeatherAsync(string city)
    {
        try
        {
            var response = await _client.GetStringAsync($"https://api.weather.com/{city}");
            return JsonSerializer.Deserialize<WeatherData>(response);
        }
        catch (HttpRequestException ex)
        {
            Console.WriteLine($"天気データの取得に失敗: {ex.Message}");
            throw;
        }
    }
}

public class WeatherData
{
    public string City { get; set; }
    public double Temperature { get; set; }
}

複数の非同期処理の制御

複数の非同期処理を実行する場合の方法です。

// すべての処理の完了を待つ
public async Task GetAllDataAsync()
{
    var task1 = GetWeatherAsync("Tokyo");
    var task2 = GetWeatherAsync("Osaka");
    var task3 = GetWeatherAsync("Fukuoka");

    var results = await Task.WhenAll(task1, task2, task3);
    foreach (var result in results)
    {
        Console.WriteLine($"{result.City}: {result.Temperature}℃");
    }
}

// 最初に完了した処理の結果を使用
public async Task GetFastestDataAsync()
{
    var task1 = GetWeatherAsync("Tokyo");
    var task2 = GetWeatherAsync("Osaka");

    var firstResult = await Task.WhenAny(task1, task2);
    var weatherData = await firstResult;
    Console.WriteLine($"最初の結果: {weatherData.City}");
}

パフォーマンス最適化

非同期プログラミングを効果的に活用することで、アプリケーションのパフォーマンスを大きく向上させることができます!

ConfigureAwaitの使用

UI スレッドへの復帰が不要な場合は、ConfigureAwait(false)を使用してパフォーマンスを改善できます。

public async Task OptimizedMethodAsync()
{
    // UI スレッドへの復帰が不要な処理
    var data = await GetDataAsync().ConfigureAwait(false);
    
    // データ処理
    var processedData = await ProcessDataAsync(data).ConfigureAwait(false);
    
    return processedData;
}

.NET Core/.NET 5+ではデフォルトでSynchronizationContextがないため、UIスレッド復帰を意識する必要は少なくなります。しかし、クロスプラットフォームや旧.NET Frameworkをサポートする場合はConfigureAwait(false)を適切に活用しましょう。

軽量な非同期処理におけるValueTask

軽量な非同期処理では、Taskの代わりにValueTaskを使用することでメモリのオーバーヘッドを削減できます。ValueTaskは即座に結果が返るケースや、頻繁に呼び出される非同期メソッドで特に有効です。

例えば、以下のようなシナリオでValueTaskを使用できます。

public async ValueTask<int> GetValueAsync()
{
    return 42; // すぐに結果が返る場合、Taskより効率的
}

テスト方法

非同期メソッドのテストは、通常のメソッドと少し異なります。

[TestFixture]
public class WeatherServiceTests
{
    private WeatherService _service;

    [SetUp]
    public void Setup()
    {
        _service = new WeatherService();
    }

    [Test]
    public async Task GetWeather_ValidCity_ReturnsWeatherData()
    {
        // Arrange
        string city = "Tokyo";

        // Act
        var result = await _service.GetWeatherAsync(city);

        // Assert
        Assert.That(result, Is.Not.Null);
        Assert.That(result.City, Is.EqualTo(city));
    }
}

まとめ

非同期プログラミングの主なポイントは以下の通りです。

  1. 時間のかかる処理は非同期で実行する
  2. async voidは避け、代わりにasync Taskを使用する
  3. awaitを忘れずに使用する
  4. 例外処理を適切に行う
  5. パフォーマンスを考慮してConfigureAwait(false)を使用する

参考資料

5
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
5
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?