如果我单击一个启动backgroundworker两次的按钮,我会收到此错误。
This BackgroundWorker is currently busy and cannot run multiple tasks concurrently
我该如何避免这种情况?
答案 0 :(得分:88)
简单:不要两次启动BackgroundWorker。
您可以使用IsBusy
属性检查它是否已在运行,因此只需更改此代码:
worker.RunWorkerAsync();
到此:
if( !worker.IsBusy )
worker.RunWorkerAsync();
else
MessageBox.Show("Can't run the worker twice!");
更新
如果您确实需要同时启动多个后台任务,则只需创建多个BackgroundWorker对象
答案 1 :(得分:53)
为要执行的每个操作创建一个新的BackgroundWorker对象。即,而不是:
BackgroundWorker worker = new BackgroundWorker();
worker.DoWork += new DoWorkEventHandler(worker_DoWork);
for (int i; i < max; i++) {
worker.RunWorkerAsync(i);
}
试试这个:
for (int i; i < max; i++) {
BackgroundWorker worker = new BackgroundWorker();
worker.DoWork += new DoWorkEventHandler(worker_DoWork);
worker.RunWorkerAsync(i);
}
答案 2 :(得分:6)
我会考虑排队需要完成的任务。您将获得以下优势;
以下是一个示例实现:http://thevalerios.net/matt/2008/05/a-queued-backgroundworker。我不确定是否在线程安全中实现,一旦我弄清楚我正在使用的实现中的当前锁定问题,我将更新我的答案。
答案 3 :(得分:1)
虽然不是最初由OP提出的情况,但如果您在某种生产者中使用背景工作者,这也可能由于竞争条件(现在发生在我身上并寻找答案)而发生 - 消费者模式。
示例:
if (BckgrndWrkr == null)
{
BckgrndWrkr = new BackgroundWorker();
BckgrndWrkr.DoWork += DoWorkMethod;
BckgrndWrkr.RunWorkerAsync();
}
else if (!BckgrndWrkr.IsBusy)
{
BckgrndWrkr.RunWorkerAsync();
}
在这种情况下,有一个竞争条件:第一个实例实例化一个新的后台工作程序,第二个实例到达else if
并启动后台工作程序,然后第一个实例到达if
的RunWorkerAsync阻止,并在它发生时抛出错误。
可以通过向整个if + if else部分添加锁定来避免这种情况。