當使用子執行緒(Other Thread)更新主執行緒所顯示的UI元件時,通常會遇到「跨執行緒作業無效: 存取控制項 ... 時所使用的執行緒與建立控制項的執行緒不同」這個錯誤,可以利用UI.InvokeRequired來進行判斷跨執行緒的問題與導正回主執行緒。
在多執行緒(線程)編程中,我們經常要在工作執行緒(線程)中去更新介面顯示,而在多執行緒(線程)中直接調用介面控制元件的方法是錯誤的做法,Invoke 和 BeginInvoke 就是為了解決這個問題而出現的,使你在多執行緒(線程)中安全的更新介面顯示。
正確的做法是將工作執行緒(線程)中涉及更新介面的程式碼封裝為一個方法(method),通過 Invoke 或者 BeginInvoke 去呼叫,兩者的區別就是一個導致工作執行緒(線程)等待,而另外一個則不會。
而所謂的"一面響應操作,一面添加節點" 永遠只能是相對的,使 UI 執行緒(線程)的負擔不至於太大而已,因為介面的正確更新始終要通過 UI執行緒(線程)去做,我們要做的事情是在工作執行緒(線程)中囊括大部分的運算,而將對純粹的介面更新放到 UI執行緒(線程)中去做,這樣也就達到了減輕 UI 執行緒(線程)負擔的目的了。
以下例子說明下使用方法method,比如你在呼叫一個執行緒(線程),在執行緒(線程)的方法中想更新Form中的一個TextBox.
using System.Threading;
//啟動一個執行緒(線程)
Thread thread=new Thread(new ThreadStart(DoWork));
thread.Start();
//執行緒(線程)方法
private void DoWork()
{
this.TextBox1.Text="我是一個Textbox";
}
如果你像上面操作,在VS2005或2008裡是會有異常的...
正確的做法是用Invoke/BeginInvoke
using System.Threading;
namespace test
{
public partial class Form1 : Form
{
public delegate void MyInvoke(string str1,string str2);
public Form1()
{
InitializeComponent();
}
public void DoWork()
{
MyInvoke mi = new MyInvoke(UpdateForm);
this.BeginInvoke(mi, new Object[] {"我是Textbox","haha"});
}
public void UpdateForm(string param1,string parm2)
{
this.textBox1.Text = param1+parm2;
}
private void button1_Click(object sender, EventArgs e)
{
Thread thread = new Thread(new ThreadStart(DoWork));
thread.Start();
}
}
}
->注意委派(代理)的使用!
基本上delegate(委派)是一種型別安全(Type-Safe)的函式指標,他可以對於某個方法進行同步叫用,但對於獨立的執行緒中他是以非同步的方法來叫用。至於 Invoke則是用來叫用委派的方法,委派另外還提供了兩個呼叫非同步的方法BeginInvoke及EndInvoke,其中BeginInvoke又 會傳回一個IAsyncResult。
至於程式碼您可以參考一下MSDN文件庫:(裡面有範例程式碼及詳細說明)
事件和委派
http://msdn.microsoft.com/zh-tw/library/17sde2xt(VS.80).aspx
在建立控制項基礎控制代碼的執行緒上執行指定的非同步委派。
http://msdn.microsoft.com/zh-tw/library/0b1bf3y3(VS.85).aspx
IAsyncResult 介面
http://msdn.microsoft.com/zh-tw/library/system.iasyncresult.aspx
reference:
http://www.blueshop.com.tw/board/show.asp?subcde=BRD20080707220620NQ1&fumcde=FUM20050124192253INM&rplcnt=3
http://tw.myblog.yahoo.com/jw!tn0RjAyfEQWScG0DOCgG/article?mid=100&prev=101&next=98
http://programming.im.ncnu.edu.tw/J_Chapter9.htm