简单使用BackgroundWorker创建多个线程的教程
backgroundworker是一个非常不错的线程控件,能避免界面假死,让线程操作你想要做的事,它学习起来很简单,但是能实现很强大的功能。发布这篇文章的目的是将最近学习到的共享出来,大家交流一下,当然我也是菜鸟,在这里你将学习到backgroundworker简单使用,停止,暂停,继续等操作,backgroundworker比起thread和threadpool要简单太多,为了更方便在实际应用中使用,我使用的是winform,没有使用控制台程序。
在ui界面里拖动一个button和richtextbox到界面。
我会从最简单的开始,只有最简单的代码才会让人有继续学下去的欲望,下列代码可以将1到999打印到richtextbox1控件上。
private void button1_click(object sender, eventargs e)
{
//创建一个backgroundworker线程
backgroundworker bw = new backgroundworker();
//创建一个dowork事件,指定bw_dowork方法去做事
bw.dowork += new doworkeventhandler(bw_dowork);
//开始执行
bw.runworkerasync();
}
void bw_dowork(object sender, doworkeventargs e)
{
for (int i = 0; i < 1000; i++)
{
this.richtextbox1.text += i + environment.newline;
}
}
但是很不幸,以上代码会报错,报错信息:线程间操作无效: 从不是创建控件“richtextbox1”的线程访问它。
那么我们继续改造代码,让数字显示在richtextbox1控件上,并且让richtextbox1焦点处于最低端。
private void button1_click(object sender, eventargs e)
{
//创建一个backgroundworker线程
backgroundworker bw = new backgroundworker();
//创建一个dowork事件,指定bw_dowork方法去做事
bw.dowork += new doworkeventhandler(bw_dowork);
//开始执行
bw.runworkerasync();
}
void bw_dowork(object sender, doworkeventargs e)
{
for (int i = 0; i < 1000; i++)
{
this.invoke((methodinvoker)delegate
{
this.richtextbox1.text += i + environment.newline;
});
}
}
private void richtextbox1_textchanged(object sender, eventargs e)
{
richtextbox textbox = (richtextbox)sender;
textbox.selectionstart = textbox.text.length;
textbox.scrolltocaret();
}
上面是backgroundworker一个最简单的例子,没有多余复杂的代码,这就是backgroundworker,下面我们加入停止按钮,让线程停下来。
再拖动一个button控件到界面,让线程停止我们先要改造一下代码,让button事件也能控制到backgroundworker线程。
backgroundworker bw = null;
private void button1_click(object sender, eventargs e)
{
//创建一个backgroundworker线程
bw = new backgroundworker();
//指定可以让线程停止
bw.workersupportscancellation = true;
//创建一个dowork事件,指定bw_dowork方法去做事
bw.dowork += new doworkeventhandler(bw_dowork);
//开始执行
bw.runworkerasync();
}
private void button2_click(object sender, eventargs e)
{
//停止线程
bw.cancelasync();
}
void bw_dowork(object sender, doworkeventargs e)
{
for (int i = 0; i < 1000; i++)
{
//获取当前线程是否得到停止的指令
if (bw.cancellationpending)
{
e.cancel = true;
return;
}
this.invoke((methodinvoker)delegate
{
this.richtextbox1.text += i + environment.newline;
});
}
}
为了避免代码的复杂化,上面代码我没有做更多的体验修改,比如点击开始的按钮,开始的按钮应该为不可用状态,点击停止按钮后停止按钮不可用状态,激活开始按钮。
下面我们将继续升级,如何来获知线程是否已经执行完成或者线程已经停止了呢
backgroundworker bw = null;
private void button1_click(object sender, eventargs e)
{
bw = new backgroundworker();
bw.workersupportscancellation = true;
bw.dowork += new doworkeventhandler(bw_dowork);
//线程完成或者停止发生的事件
bw.runworkercompleted += new runworkercompletedeventhandler(bw_runworkercompleted);
bw.runworkerasync();
}
private void button2_click(object sender, eventargs e)
{
bw.cancelasync();
}
void bw_dowork(object sender, doworkeventargs e)
{
for (int i = 0; i < 1000; i++)
{
if (bw.cancellationpending)
{
e.cancel = true;
return;
}
this.invoke((methodinvoker)delegate
{
this.richtextbox1.text += i + environment.newline;
});
}
}
void bw_runworkercompleted(object sender, runworkercompletedeventargs e)
{
if (e.cancelled)
{
this.richtextbox1.text += "线程已经停止";
}
else
{
this.richtextbox1.text += "线程已经完成";
}
}
到现在为止你可以自己去用backgroundworker创建一个线程了,你已经了解它了,当然backgroundworker还有一个reportprogress滚动条事件,可以显示进度,我暂且认为它是多余的,因为大部分进度都可以通过bw_dowork来控制实现。下面我们继续完善backgroundworker,加入暂停和继续功能。
再拖动一个button控件到界面,backgroundworker的暂停和继续我们使用manualresetevent。
backgroundworker bw = null;
//创建manualresetevent
manualresetevent mr = new manualresetevent(true);
private void button1_click(object sender, eventargs e)
{
bw = new backgroundworker();
bw.workersupportscancellation = true;
bw.dowork += new doworkeventhandler(bw_dowork);
bw.runworkercompleted += new runworkercompletedeventhandler(bw_runworkercompleted);
bw.runworkerasync();
}
private void button2_click(object sender, eventargs e)
{
bw.cancelasync();
}
private void button3_click(object sender, eventargs e)
{
button b = (button)sender;
if (b.text == "暂停")
{
mr.reset();
b.text = "继续";
}
else
{
mr.set();
b.text = "暂停";
}
}
void bw_dowork(object sender, doworkeventargs e)
{
for (int i = 0; i < 1000; i++)
{
if (bw.cancellationpending)
{
e.cancel = true;
return;
}
this.invoke((methodinvoker)delegate
{
this.richtextbox1.text += i + environment.newline;
});
//接受指令
mr.waitone();
}
}
void bw_runworkercompleted(object sender, runworkercompletedeventargs e)
{
if (e.cancelled)
{
this.richtextbox1.text += "线程已经停止";
}
else
{
this.richtextbox1.text += "线程已经完成";
}
}
到目前为止backgroundworker的大部分功能都实现了,上面的代码在很多博客中都能找到,都是只执行了一个后台线程。如果我们有1千个耗时的任务,那么一个线程远远不够,我们需要创建多条线程,让他分段执行,比如创建10个线程,把1千个任务分成不同的等分让10个线程分别去执行。
我们使用list泛型 list<backgroundworker>,然后使用bw.runworkerasync(i) 传递参数到bw_dowork里,在bw_dowork里使用e.argument接受参数。
list<backgroundworker> bws = new list<backgroundworker>();
int t = 10;
private void button1_click(object sender, eventargs e)
{
for (int i = 0; i < t; i++)
{
backgroundworker bw = new backgroundworker();
bw.dowork += new doworkeventhandler(bw_dowork);
bws.add(bw);
bw.runworkerasync(i);
}
}
void bw_dowork(object sender, doworkeventargs e)
{
int j = convert.toint32(e.argument);
for (int i = j; i < 1000; i = i + t)
{
if (((backgroundworker)sender).cancellationpending)
{
e.cancel = true;
return;
}
string item = string.format("线程{0}正在操作数据{1}", j + 1, i);
this.invoke((methodinvoker)delegate
{
this.richtextbox1.text += item + environment.newline;
});
//thread.sleep(200);
}
}
由于上面代码不是耗时操作,又开启线程10个,操作过快,造成界面假死状态,可以使用sleep让线程休眠。
我们继续完善代码,加入停止操作,加入完成后和停止的事件,由于是多线程,判断是线程操作是否完成,我们用bws.remove(sender as backgroundworker); 方法删除线程,然后使用bws.count == 0来判断是否操作完成。
list<backgroundworker> bws = new list<backgroundworker>();
int t = 10;
private void button1_click(object sender, eventargs e)
{
for (int i = 0; i < t; i++)
{
backgroundworker bw = new backgroundworker();
bw.dowork += new doworkeventhandler(bw_dowork);
bw.workersupportscancellation = true;
bw.runworkercompleted += new runworkercompletedeventhandler(bw_runworkercompleted);
bws.add(bw);
bw.runworkerasync(i);
}
}
private void button2_click(object sender, eventargs e)
{
for (int i = 0; i < t; i++)
{
bws[i].cancelasync();
}
}
void bw_dowork(object sender, doworkeventargs e)
{
int j = convert.toint32(e.argument);
for (int i = j; i < 1000; i = i + t)
{
if (((backgroundworker)sender).cancellationpending)
{
e.cancel = true;
return;
}
string item = string.format("线程{0}正在操作数据{1}", j + 1, i);
this.invoke((methodinvoker)delegate
{
this.richtextbox1.text += item + environment.newline;
});
thread.sleep(200);
}
}
void bw_runworkercompleted(object sender, runworkercompletedeventargs e)
{
bws.remove(sender as backgroundworker);
if (bws.count == 0)
{
if (e.cancelled)
{
this.richtextbox1.text += "线程已经停止";
}
else
{
this.richtextbox1.text += "线程已经完成";
}
}
}
上面代码中的停止不是能立即停止,这个就和开车一样,开的越快,刹车的后拖行的距离越长,同理,开启的线程的越多,完全暂停需要的时间越长,不知我说的是否正确。另外我也想问一下是否能真正的全部线程停止,点停止后全部线程立即停止。
下面我们继续加入暂停和继续的功能,一样的道理,我们使用list<manualresetevent>。
list<backgroundworker> bws = new list<backgroundworker>();
list<manualresetevent> mrs = new list<manualresetevent>();
int t = 10;
private void button1_click(object sender, eventargs e)
{
for (int i = 0; i < t; i++)
{
backgroundworker bw = new backgroundworker();
bw.dowork += new doworkeventhandler(bw_dowork);
bw.workersupportscancellation = true;
bw.runworkercompleted += new runworkercompletedeventhandler(bw_runworkercompleted);
bws.add(bw);
bw.runworkerasync(i);
mrs.add(new manualresetevent(true));
}
}
private void button2_click(object sender, eventargs e)
{
for (int i = 0; i < t; i++)
{
bws[i].cancelasync();
}
}
private void button3_click(object sender, eventargs e)
{
button b = (button)sender;
if (b.text == "暂停")
{
for (int i = 0; i < mrs.count; i++)
{
mrs[i].reset();
}
b.text = "继续";
}
else
{
for (int i = 0; i < mrs.count; i++)
{
mrs[i].set();
}
b.text = "暂停";
}
}
void bw_dowork(object sender, doworkeventargs e)
{
int j = convert.toint32(e.argument);
for (int i = j; i < 1000; i = i + t)
{
if (((backgroundworker)sender).cancellationpending)
{
e.cancel = true;
return;
}
string item = string.format("线程{0}正在操作数据{1}", j + 1, i);
this.invoke((methodinvoker)delegate
{
this.richtextbox1.text += item + environment.newline;
});
thread.sleep(200);
mrs[j].waitone();
}
}
void bw_runworkercompleted(object sender, runworkercompletedeventargs e)
{
bws.remove(sender as backgroundworker);
if (bws.count == 0)
{
if (e.cancelled)
{
this.richtextbox1.text += "线程已经停止";
}
else
{
this.richtextbox1.text += "线程已经完成";
}
}
}
至此,所有的代码都奉上了,多个线程操作会带来很多意向不到的麻烦,比如多个线程同时把数据写入一个文件,多线程更新datatable等,会让软件莫名其妙的自动退出,.net2.0里还没有绝对线程安全的数据集,很多大佬都说用lock,但我对lock也是一知半解,还请大家赐教赐教,如上有什么说的不对,也请大家多多指点。
推荐阅读
-
简单使用BackgroundWorker创建多个线程的教程
-
简单使用BackgroundWorker创建多个线程的教程
-
中文Access2000速成教程--2.1 使用向导创建简单的选择查询
-
Java线程池的简单使用方法实例教程
-
C#使用Windows Service的简单教程(创建、安装、卸载、调试)
-
中文Access2000速成教程--2.1 使用向导创建简单的选择查询
-
中文Access2000速成教程--2.1 使用向导创建简单的选择查询
-
Java线程池的简单使用方法实例教程
-
使用QTcpServer创建一个简单的服务器程序之一:单线程服务器
-
C#使用Windows Service的简单教程(创建、安装、卸载、调试)