使用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),
              ("http:////" + (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),
              ("http:////" + (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),
              ("http:////" + (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()
個人感覺語法較麻煩,尤其是對每種界面修改都必須定義一種委托(因為參數不同),有辦法改進么?
新聞熱點
疑難解答