winform程序中如何跨线程修改控件的值

时间:2022-03-25 20:17:24

winform程序是单线程的。

        /// <summary>
/// 应用程序入口
/// </summary>
[STAThread]
static void Main()

而且对某一个控件来说,只有创建该控件的线程才能修改它的值。比如我们在设计器中拖到窗体上的控件,它们由程序的主线程创建,那么如果我们在执行中又创建了另外一个线程,那么我们在这个新创建的线程中无法直接修改窗体上控件的值。

winform程序中如何跨线程修改控件的值

有时候我们的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);
}