繁体   English   中英

C# - 后台工作者的CancelAsync()不起作用?

[英]C#- background worker's CancelAsync() not working?

我想中止该过程但不能这样做,我正在使用后台工作程序和我的处理功能。

public void Init()
{
    bw = new BackgroundWorker();
    bw.WorkerSupportsCancellation = true;
    bw.DoWork += new DoWorkEventHandler(bw_DoWork);
    bw.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bw_RunWorkerCompleted);
}

void bw_DoWork(object sender, DoWorkEventArgs e)
{
    if (bw.CancellationPending == true)
    {
        e.Cancel = true;
    }
    else
    {
        e.Result = abd();
    }
}

void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
{
    if(e.Cancelled)
    {
        lbltext.content="Canceled";
    }

    else
    {
        lbltext.content="Completed";
    }
}

private void btncan_Click(object sender, RoutedEventArgs e)
{
    bw.CancelAsync();
}

private void btnstart_Click(object sender, RoutedEventArgs e)
{
    bw.RunWorkerAsync();
}

我无法使用此代码中止该过程。 函数abd()正在执行处理部分并返回结果。

请给我任何解决方案。

谢谢。

当你调用bw.CancelAsync()你只需将CancellationPending标志设置为true 默认情况下它不会取消某些内容。 您需要手动处理待处理的取消。 但是你不能用你的代码做到这一点,因为当你点击按钮时,有三种可能的选择:

  • 长时间运行的abd()方法完成了它的工作,没有什么可以取消
  • abd()启动它的工作,后台工作程序被阻塞 - 它正在等待abd()结果,然后它继续执行 - 即退出if-else阻塞并引发RunWorkerCompleted事件。
  • 几乎不可能的选项 - 你将快速点亮,你会在输入if-else块之前点击按钮。 CancellationPending为真, abd()不会开始执行

如果要使用取消,请在循环中执行长时间运行的任务,并检查每个步骤是否有待取消:

void bw_DoWork(object sender, DoWorkEventArgs e)
{
    List<Foo> results = new List<Foo>();

    // any loop here - foreach, while
    for(int i = 0; i < steps_count; i++)
    {    
         // check status on each step
         if (bw.CancellationPending == true) 
         {
             e.Cancel = true;
             return; // abort work, if it's cancelled
         }

         results.Add(abd()); // add part of results
    }

    e.Result = results; // return all results
}

可能DoWork可能在调用CancelAsync之前完成了它的工作,并且如文档中提到的那样e.Cancelled可能是错误的。

文件说这个

请注意,DoWork事件处理程序中的代码可能会在取消请求时完成其工作,并且您的轮询循环可能会错过CancellationPending设置为true。 在这种情况下,即使发出了取消请求,RunWorkerCompleted事件处理程序中System.ComponentModel.RunWorkerCompletedEventArgs的Canceled标志也不会设置为true。 这种情况称为竞争条件,是多线程编程中的常见问题。

以下怎么样?

While(!bw.CancellationPending)
{
   //do some work!
}
e.Cancel = true;

看起来像函数abd()需要花费太多的处理时间而你想在它们之间中止它。

private Thread _backgroundWorkerThread;

public void AbortBackgroundWorker()
{
    if(_backgroundWorkerThread != null)
    _backgroundWorkerThread.Abort();
}

void DoWork(object sender, DoWorkEventArgs e)
{
    try
    {
    _backgroundWorkerThread = Thread.CurrentThread;

    // call abd...

    }
    catch(ThreadAbortException)
    {
        // Do your clean up here.
    }
}

AbortBackgroundWorker()放在btncan_Click事件中。 但这带来了成本。 现在RunWorkerCompleted事件将不起作用。 所以,你必须在catch块中处理它或者写一些我们自己的代码。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM