C# BackgroundWorker 控件使用Word文件下载.docx
- 文档编号:3675386
- 上传时间:2023-05-02
- 格式:DOCX
- 页数:16
- 大小:17.44KB
C# BackgroundWorker 控件使用Word文件下载.docx
《C# BackgroundWorker 控件使用Word文件下载.docx》由会员分享,可在线阅读,更多相关《C# BackgroundWorker 控件使用Word文件下载.docx(16页珍藏版)》请在冰点文库上搜索。
//创建一个DoWork事件,指定bw_DoWork方法去做事
bw.DoWork+=newDoWorkEventHandler(bw_DoWork);
//开始执行
bw.RunWorkerAsync();
}
voidbw_DoWork(objectsender,DoWorkEventArgse)
for(inti=0;
i<
1000;
i++)
this.richTextBox1.Text+=i+Environment.NewLine;
但是很不幸,以上代码会报错,报错信息:
线程间操作无效:
从不是创建控件“richTextBox1”的线程访问它。
那么我们继续改造代码,让数字显示在richTextBox1控件上,并且让richTextBox1焦点处于最低端。
this.Invoke((MethodInvoker)delegate
});
privatevoidrichTextBox1_TextChanged(objectsender,EventArgse)
RichTextBoxtextbox=(RichTextBox)sender;
textbox.SelectionStart=textbox.Text.Length;
textbox.ScrollToCaret();
上面是BackgroundWorker一个最简单的例子,没有多余复杂的代码,这就是BackgroundWorker,下面我们加入停止按钮,让线程停下来。
再拖动一个button控件到界面,让线程停止我们先要改造一下代码,让button事件也能控制到BackgroundWorker线程。
BackgroundWorkerbw=null;
privatevoidbutton1_Click(objectsender,EventArgse)
bw=newBackgroundWorker();
//指定可以让线程停止
bw.WorkerSupportsCancellation=true;
privatevoidbutton2_Click(objectsender,EventArgse)
//停止线程
bw.CancelAsync();
//获取当前线程是否得到停止的指令
if(bw.CancellationPending)
e.Cancel=true;
return;
为了避免代码的复杂化,上面代码我没有做更多的体验修改,比如点击开始的按钮,开始的按钮应该为不可用状态,点击停止按钮后停止按钮不可用状态,激活开始按钮。
下面我们将继续升级,如何来获知线程是否已经执行完成或者线程已经停止了呢
//线程完成或者停止发生的事件
bw.RunWorkerCompleted+=newRunWorkerCompletedEventHandler(bw_RunWorkerCompleted);
voidbw_RunWorkerCompleted(objectsender,RunWorkerCompletedEventArgse)
if(e.Cancelled)
this.richTextBox1.Text+="
线程已经停止"
;
else
线程已经完成"
到现在为止你可以自己去用BackgroundWorker创建一个线程了,你已经了解它了,当然BackgroundWorker还有一个ReportProgress滚动条事件,可以显示进度,我暂且认为它是多余的,因为大部分进度都可以通过bw_DoWork来控制实现。
下面我们继续完善BackgroundWorker,加入暂停和继续功能。
再拖动一个button控件到界面,BackgroundWorker的暂停和继续我们使用ManualResetEvent。
//创建ManualResetEvent
ManualResetEventmr=newManualResetEvent(true);
privatevoidbutton3_Click(objectsender,EventArgse)
Buttonb=(Button)sender;
if(b.Text=="
暂停"
)
{
mr.Reset();
b.Text="
继续"
}
else
mr.Set();
//接受指令
mr.WaitOne();
到目前为止BackgroundWorker的大部分功能都实现了,上面的代码在很多博客中都能找到,都是只执行了一个后台线程。
如果我们有1千个耗时的任务,那么一个线程远远不够,我们需要创建多条线程,让他分段执行,比如创建10个线程,把1千个任务分成不同的等分让10个线程分别去执行。
我们使用list泛型List<
BackgroundWorker>
,然后使用bw.RunWorkerAsync(i)传递参数到bw_DoWork里,在bw_DoWork里使用e.Argument接受参数。
List<
bws=newList<
();
intt=10;
t;
bws.Add(bw);
bw.RunWorkerAsync(i);
intj=Convert.ToInt32(e.Argument);
for(inti=j;
i=i+t)
if(((BackgroundWorker)sender).CancellationPending)
stringitem=String.Format("
线程{0}正在操作数据{1}"
j+1,i);
this.richTextBox1.Text+=item+Environment.NewLine;
//Thread.Sleep(200);
由于上面代码不是耗时操作,又开启线程10个,操作过快,造成界面假死状态,可以使用Sleep让线程休眠。
我们继续完善代码,加入停止操作,加入完成后和停止的事件,由于是多线程,判断是线程操作是否完成,我们用bws.Remove(senderasBackgroundWorker);
方法删除线程,然后使用bws.Count==0来判断是否操作完成。
bws[i].CancelAsync();
Thread.Sleep(200);
bws.Remove(senderasBackgroundWorker);
if(bws.Count==0)
上面代码中的停止不是能立即停止,这个就和开车一样,开的越快,刹车的后拖行的距离越长,同理,开启的线程的越多,完全暂停需要的时间越长,不知我说的是否正确。
另外我也想问一下是否能真正的全部线程停止,点停止后全部线程立即停止。
下面我们继续加入暂停和继续的功能,一样的道理,我们使用List<
ManualResetEvent>
。
List<
mrs=newList<
mrs.Add(newManualResetEvent(true));
mrs.Count;
i++)
mrs[i].Reset();
mrs[i].Set();
mrs[j].WaitOne();
至此,所有的代码都奉上了,多个线程操作会带来很多意向不到的麻烦,比如多个线程同时把数据写入一个文件,多线程更新datatable等,会让软件莫名其妙的自动退出,.net2.0里还没有绝对线程安全的数据集,很多大佬都说用lock,但我对lock也是一知半解,还请大家赐教赐教,如上有什么说的不对,也请大家多多指点。
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- C# BackgroundWorker 控件使用 控件 使用