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