c#2005后不再支持多线程直接访问界面的控件(界面创建线程与访问线程不是同一个线程),不过可以使用delegate来解决:
1. 声明一个delegate和定义一个delegate的实现函数
view plaincopy to clipboardprint?
delegate void showprogressdelegate(int newpos);
private void showprogress(int newpos)
{
// 判断是否在线程中访问
if (!_progressbar.invokerequired)
{
// 不是的话直接操作控件
_progressbar.value = newpos;
}
else
{
// 是的话启用delegate访问
showprogressdelegate showprogress = new showprogressdelegate(showprogress);
// 如使用invoke会等到函数调用结束,而begininvoke不会等待直接往后走
this.begininvoke(showprogress, new object[] { newpos });
}
}
delegate void showprogressdelegate(int newpos);
private void showprogress(int newpos)
{
// 判断是否在线程中访问
if (!_progressbar.invokerequired)
{
// 不是的话直接操作控件
_progressbar.value = newpos;
}
else
{
// 是的话启用delegate访问
showprogressdelegate showprogress = new showprogressdelegate(showprogress);
// 如使用invoke会等到函数调用结束,而begininvoke不会等待直接往后走
this.begininvoke(showprogress, new object[] { newpos });
}
}
2. 定义线程函数(在另一个线程中可以对界面控件进读操作)
view plaincopy to clipboardprint?
private void progressstart()
{
while (true)
{
int newpos = _progressbar.value + 10;
if (newpos > _progressbar.maximum)
{
newpos = _progressbar.minimum;
}
trace.writeline(string.format("pos: {0}", newpos));
// 这里直接调用方法,由其内部自动判断是否启用delegate
showprogress(newpos);
thread.sleep(100);
}
}
private void progressstart()
{
while (true)
{
int newpos = _progressbar.value + 10;
if (newpos > _progressbar.maximum)
{
newpos = _progressbar.minimum;
}
trace.writeline(string.format("pos: {0}", newpos));
// 这里直接调用方法,由其内部自动判断是否启用delegate
showprogress(newpos);
thread.sleep(100);
}
}
3. 线程的启动和终止
view plaincopy to clipboardprint?
private thread _progressthread;
_progressthread = new thread(new threadstart(progressstart));
// 可选,功用:即使该线程不结束,进程也可以结束
_progressthread.isbackground = true;
_progressthread.start();
_progressthread.abort();
// 可选,功用:等到线程结束才继续
_progressthread.join();
_progressthread = null;
以上就是c#多线程访问界面的内容,更多相关内容请关注PHP中文网(www.php.cn)!
