如何正确重置 CancellationToken?

Posted

技术标签:

【中文标题】如何正确重置 CancellationToken?【英文标题】:How to reset a CancellationToken properly? 【发布时间】:2012-03-09 02:33:28 【问题描述】:

今天早上我一直在玩Async CTP,并有一个简单的程序,其中包含buttonlabel。单击button,它开始更新label,停止button,它停止写入label。但是,我不确定如何重置 CancellationTokenSource 以便我可以重新启动该过程。

我的代码:

public partial class MainWindow : Window

    CancellationTokenSource cts = new CancellationTokenSource();
    public MainWindow()
    
        InitializeComponent();
        button.Content = "Start";
    

    async Task DoWork(CancellationToken cancelToken)
    
        int i = 0;
        while (!cancelToken.IsCancellationRequested)
        
            label.Content = i++.ToString();
            await TaskEx.Delay(50, cancelToken);
        
    

    private void Button_Click(object sender, RoutedEventArgs e)
    
        if (button.Content == "Start")
        
            button.Content = "Stop";
            DoWork(cts.Token);
        
        else
        
            button.Content = "Start";
            cts.Cancel();
        
    

【问题讨论】:

【参考方案1】:

您需要重新创建 CancellationTokenSource - 设置后无法“重置”它。

这可能很简单:

private void Button_Click(object sender, RoutedEventArgs e)

    if (button.Content == "Start")
    
        button.Content = "Stop";
        cts.Dispose(); // Clean up old token source....
        cts = new CancellationTokenSource(); // "Reset" the cancellation token source...
        DoWork(cts.Token);
    
    else
    
        button.Content = "Start";
        cts.Cancel();
    

【讨论】:

你应该在什么时候在应用程序关闭时处理它?因为你必须在线程完成之前等待,否则你会得到 ObjectDisposed 异常。 @user2587718 这真的取决于什么类型的对象等。我建议您对此提出自己的问题。 真的应该在处理之前做一个空检查,但这很好,因为这正是我正在做的。 想了解更多关于 CancellationTokens 的知识,为什么不在Cancel 通话后立即处理 cts?是否应该只在下次更新时才丢弃?【参考方案2】:

我遇到了同样的问题,我发现解决它的最佳方法是在调用该方法之前重新创建取消令牌源。

这是我点击开始按钮时所做的:

cancellationTokenSource = new CancellationTokenSource();
cancellationToken = cancellationTokenSource.Token;
Task.Factory.StartNew(StartUpload, cancellationToken);

我将同一个按钮的标题更改为取消,当点击取消时,我调用

cancellationTokenSource.Cancel();

这里是完整的代码:

if (button3.Text != "&Start Upload")

    cancellationTokenSource.Cancel();

else

    try
    
        cancellationTokenSource = new CancellationTokenSource();
        cancellationToken = cancellationTokenSource.Token;
        Task.Factory.StartNew(StartUpload, cancellationToken);
    
    catch (AggregateException ex)
    
        var builder = new StringBuilder();
        foreach (var v in ex.InnerExceptions)
            builder.Append("\r\n" + v.InnerException);
        MessageBox.Show("There was an exception:\r\n" + builder.ToString());
    
    catch (Exception ex)
    
        MessageBox.Show(ex.Message);
    

【讨论】:

【参考方案3】:

现在,在 .NET 6 中,您可以使用 TryReset() 方法,Attempts to reset the CancellationTokenSource to be used for an unrelated operation. 请参阅 this 问题和 CancellationTokenSource.cs 了解更多详细信息。

【讨论】:

以上是关于如何正确重置 CancellationToken?的主要内容,如果未能解决你的问题,请参考以下文章

如何取消 CancellationToken

如何使用 CancellationToken 属性?

如何强制 IAsyncEnumerable 尊重 CancellationToken

C# CancellationToken 如何与 SqlConnection.OpenAsync(token) 一起使用?

如何使用 ContinueWith 正确管理任务中的异常

Firebase Auth 密码重置:如何正确发送包含确认码的密码重置电子邮件?