使用control.invoke处理多线程应用程序界面
使用单一线程,在进行大计算量或耗时的操作时,会使界面失去响应。control.invoke 提供了一个在工作线程中更新见面的简单办法。该函数会把作为参数的回调函数提交给应用程序的界面进程(一般是主线程)的队列中,等待处理。这样,对界面的操作便无须担心同步、互锁等问题。
以下例子摘自一个局域网资源搜索程序
可以看到,线程池中的工作线程通过调用treeview的invoke方法,并提供一个回调函数,来实现对treeview的更新。
在frameworks 1.1 中,部分简单的多线程更新界面并不需要这种操作,如修改一些静态文本框中的文本等。但在frameworks 2.0,所有的操作都必须要以此方式完成。
private void searchnet()
{
//清除所有根节点下的字节点
treeview1.nodes[0].nodes.clear();
for (uint i = ipstart; i <= ipend; i++)
{
//把整数转化ip地址,添加任务到线程池
threadpool.queueuserworkitem(new waitcallback(search),
("////" + (i >> 24).tostring() + '.' +
(((int) i & 0x00ff0000) >> 16).tostring() + '.' +
(((int) i & 0x0000ff00u) >> 8).tostring() + '.' + (i%256).tostring()));
}
}
delegate void updater(treenode parent, treenode child);
public void updatetreeview(treenode parent, treenode child)
{
parent.nodes.add(child);
treeview1.nodes[0].expand();
foreach (treenode n in treeview1.nodes[0].nodes)
{
n.expand();
}
}
private void search(object host)
{
wksta_info_100 stainfo;
//尝试连接
if(connect(host,out stainfo))
{
treenode nodecomputer = new treenode(stainfo.wki100_computername);
//搜索共享
searchserver(nodecomputer,stainfo, (string)host);
}
}
private void searchserver(treenode nodecomputer,wksta_info_100 stainfo, string host)
{
/*其它代码*/
treeview1.invoke(new updater(updatetreeview), new object[] {nodecomputer, node});
/*其它代码*/
}
private void searchnet()
{
//清除所有根节点下的字节点
treeview1.nodes[0].nodes.clear();
for (uint i = ipstart; i <= ipend; i++)
{
//把整数转化ip地址,添加任务到线程池
threadpool.queueuserworkitem(new waitcallback(search),
("////" + (i >> 24).tostring() + '.' +
(((int) i & 0x00ff0000) >> 16).tostring() + '.' +
(((int) i & 0x0000ff00u) >> 8).tostring() + '.' + (i%256).tostring()));
}
}
delegate void updater(treenode parent, treenode child);
public void updatetreeview(treenode parent, treenode child)
{
parent.nodes.add(child);
treeview1.nodes[0].expand();
foreach (treenode n in treeview1.nodes[0].nodes)
{
n.expand();
}
}
private void search(object host)
{
wksta_info_100 stainfo;
//尝试连接
if(connect(host,out stainfo))
{
treenode nodecomputer = new treenode(stainfo.wki100_computername);
//搜索共享
searchserver(nodecomputer,stainfo, (string)host);
}
}
private void searchserver(treenode nodecomputer,wksta_info_100 stainfo, string host)
{
/*其它代码*/
treeview1.invoke(new updater(updatetreeview), new object[] {nodecomputer, node});
/*其它代码*/
}
private void searchnet()
{
//清除所有根节点下的字节点
treeview1.nodes[0].nodes.clear();
for (uint i = ipstart; i <= ipend; i++)
{
//把整数转化ip地址,添加任务到线程池
threadpool.queueuserworkitem(new waitcallback(search),
("////" + (i >> 24).tostring() + '.' +
(((int) i & 0x00ff0000) >> 16).tostring() + '.' +
(((int) i & 0x0000ff00u) >> 8).tostring() + '.' + (i%256).tostring()));
}
}
delegate void updater(treenode parent, treenode child);
public void updatetreeview(treenode parent, treenode child)
{
parent.nodes.add(child);
treeview1.nodes[0].expand();
foreach (treenode n in treeview1.nodes[0].nodes)
{
n.expand();
}
}
private void search(object host)
{
wksta_info_100 stainfo;
//尝试连接
if(connect(host,out stainfo))
{
treenode nodecomputer = new treenode(stainfo.wki100_computername);
//搜索共享
searchserver(nodecomputer,stainfo, (string)host);
}
}
private void searchserver(treenode nodecomputer,wksta_info_100 stainfo, string host)
{
/*其它代码*/
treeview1.invoke(new updater(updatetreeview), new object[] {nodecomputer, node});
/*其它代码*/
}
总结如下:
1、 定义委托
2、 定义回调函数
3、 调用control.invoke()
个人感觉语法较麻烦,尤其是对每种界面修改都必须定义一种委托(因为参数不同),有办法改进么?
新闻热点
疑难解答