使用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()
个人感觉语法较麻烦,尤其是对每种界面修改都必须定义一种委托(因为参数不同),有办法改进么?