再試行ロジックを記述する最もクリーンな方法は? 質問する

再試行ロジックを記述する最もクリーンな方法は? 質問する

時々、諦める前に操作を何回か再試行する必要があります。私のコードは次のようになります:

int retries = 3;
while(true) {
  try {
    DoSomething();
    break; // success!
  } catch {
    if(--retries == 0) throw;
    else Thread.Sleep(1000);
  }
}

これを次のような一般的な再試行関数に書き直したいと思います。

TryThreeTimes(DoSomething);

C# でも可能ですか? メソッドのコードは何でしょうかTryThreeTimes()?

ベストアンサー1

同じ呼び出しを単純に再試行する包括的な catch ステートメントは、一般的な例外処理メカニズムとして使用すると危険です。そうは言っても、ここでは任意のメソッドで使用できるラムダ ベースの再試行ラッパーを紹介します。柔軟性を高めるために、再試行回数と再試行タイムアウトをパラメーターとして考慮することにしました。

public static class Retry
{
    public static void Do(
        Action action,
        TimeSpan retryInterval,
        int maxAttemptCount = 3)
    {
        Do<object>(() =>
        {
            action();
            return null;
        }, retryInterval, maxAttemptCount);
    }

    public static T Do<T>(
        Func<T> action,
        TimeSpan retryInterval,
        int maxAttemptCount = 3)
    {
        var exceptions = new List<Exception>();

        for (int attempted = 0; attempted < maxAttemptCount; attempted++)
        {
            try
            {
                if (attempted > 0)
                {
                    Thread.Sleep(retryInterval);
                }
                return action();
            }
            catch (Exception ex)
            {
                exceptions.Add(ex);
            }
        }
        throw new AggregateException(exceptions);
    }
}

このユーティリティ メソッドを使用して再試行ロジックを実行できるようになりました。

Retry.Do(() => SomeFunctionThatCanFail(), TimeSpan.FromSeconds(1));

または:

Retry.Do(SomeFunctionThatCanFail, TimeSpan.FromSeconds(1));

または:

int result = Retry.Do(SomeFunctionWhichReturnsInt, TimeSpan.FromSeconds(1), 4);

あるいは、asyncオーバーロードを起こすこともできます。

おすすめ記事