CancellationTokenSource.CancelAfter 不起作用

Rob*_*ann 5 c# task linqpad

我正在尝试基于这篇文章(但带有任务)实现一些重试逻辑 编写重试逻辑的最干净的方法?

重试逻辑的想法是实现第二个任务,在给定的时间后触发取消

void Main()
{
    RetryAction(() => Sleep(), 500);
}

public static void RetryAction(Action action, int timeout)
{
    var cancelSource = new CancellationTokenSource();                
    cancelSource.CancelAfter(timeout);

    Task.Run(() => action(), cancelSource.Token);    
}

public static void Sleep() 
{
    System.Threading.Thread.Sleep(5000);
    "done".Dump();
}
Run Code Online (Sandbox Code Playgroud)

上面是一个 linqPad 片段(因此是“done”.Dump())

知道为什么 CancelAfter 永远不起作用吗?

sgm*_*ore 3

您的 Sleep 方法忽略 CancellationToken。

尝试类似的东西

public static CancellationTokenSource cancelSource ;

void Main()
{
    RetryAction(() => Sleep(), 500);
}

public static void RetryAction(Action action, int timeout)
{
     cancelSource = new CancellationTokenSource();                
     cancelSource.CancelAfter(timeout);

     Task.Run(() => action(), cancelSource.Token);    
}

public static void Sleep() 
{
    for(int i = 0 ; i< 50; i++)
    {
        "Waiting".Dump();
        System.Threading.Thread.Sleep(100);

        if (cancelSource.IsCancellationRequested)
        {
            "Cancelled".Dump();
            return;
        }
    }
    "done".Dump();
}
Run Code Online (Sandbox Code Playgroud)

  • 这是一个糟糕的做法。人们应该使用“Task.Delay”而不是“Thread.Sleep”,因为“Task.Delay”支持取消标记。 (3认同)