首页 > 开发 > 综合 > 正文

使用Control.Invoke处理多线程应用程序界面

2024-07-21 02:16:04
字体:
来源:转载
供稿:网友


使用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()



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

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表