winform程序是单线程的。
/// <summary> /// 应用程序入口 /// </summary> [STAThread] static void Main() { }
而且对某一个控件来说,只有创建该控件的线程才能修改它的值。比如我们在设计器中拖到窗体上的控件,它们由程序的主线程创建,那么如果我们在执行中又创建了另外一个线程,那么我们在这个新创建的线程中无法直接修改窗体上控件的值。
有时候我们的winform程序在某一个处理上可能要会费大量的时间,这个时候我们可能会想用另一个线程来处理这个长时间的任务,而同时我 们可以做一些其它的事情。.net里面多线程异步处理可以使用ThreadPool.QueueUserWorkItem, BackgoundWorder等就可以非常简单地实现。就像上面图中显示的那样,在点击了按钮之后,我们希望异步地执行按钮的处理程序:
private void btnDoSomething_Click(object sender, EventArgs e) { WaitCallback callBack = new WaitCallback(DoSomething); ThreadPool.QueueUserWorkItem(callBack, null); } private void DoSomething(object state) { for (int i = 0; i < 100; i++) { Thread.Sleep(1000); } }
这里用Thread.Sleep(1000)来模拟了一个长时间的任务。到这里异步的目的已经达到,但是我们希望处理的同时能报告处理的进度,好给用户一些提示。比如例子中我们要更新一个进度条,可能会用如下的代码:
for (int i = 0; i < 100; i++) { Thread.Sleep(1000); //can't do the cross-thread updating reportProgress(i, 100); } private void ReportProgress(int countFinished, int total) { progressBar1.Maximum = total; progressBar1.Value = countFinished; }
我们希望可以在异步执行的线程中修改进度条控件的值,但是事与愿违。如果运行这段程序,.net会告诉我们“只有创建该控件的线程才可以修改该控件的值”, 这是线程的安全问题。但是我们确实是需要更新进度条的值怎么办呢?我们不能跨线程直接修改控件的值,但是我们可以通知控件的创建线程我们需要对某个控件进 行修改,让控件的创建线程去帮我们更新控件。“Control.Invoke(delegate,paras object[])”可以帮我们完成这一任务。
private void btnDoSomething_Click(object sender, EventArgs e) { WaitCallback callBack = new WaitCallback(DoSomething); Object progressBar = progressBar1; ThreadPool.QueueUserWorkItem(callBack, progressBar); } private void DoSomething(object state) { for (int i = 0; i < 100; i++) { Thread.Sleep(1000); //can't do the cross-thread updating //reportProgress(i, 100); ProgressBar pbar = state as ProgressBar; if (pbar != null) pbar.Invoke(reportProgress, new object[] { i, 100 }); } }
我们调用了ProgressBar的Invoke方法通知该控件的创建线程修改它的值。Invoke方法有两个重载的版本:
public object Invoke(Delegate method); public object Invoke(Delegate method, params object[] args);
这里使用了第二个,因为我们报告进度的方法需要参数。
public delegate void ReportTest(int countFinished, int total); public FormTest() { InitializeComponent(); reportProgress = new ReportTest(ReportProgress); // also can add // reportProgress += new ReportTest(ReportProgress); }