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