1. 程式人生 > >【轉】淺述WinForm多線程編程與Control.Invoke的應用

【轉】淺述WinForm多線程編程與Control.Invoke的應用

引用 pen his .net some all urn 方法 tostring

環境:VS2008、C#3.0

在WinForm開發中,我們通常不希望當窗體上點了某個按鈕執行某個業務的時候,窗體就被卡死了,直到該業務執行完畢後才緩過來。一個最直接的方法便是使用多線程。多線程編程的方式在WinForm開發中必不可少。

本文介紹在WinForm開發中如何使用多線程,以及在線程中如何通過Control.Invoke方法返回窗體主線程執行相關操作。

-. WinForm多線程編程
1. new Thread()
新開一個線程,執行一個方法,沒有參數傳遞:

private void DoWork() {
Thread t = new Thread(new ThreadStart(this.DoSomething));

t.Start();
}
private void DoSomething() {
MessageBox.Show("thread start");
}

新開一個線程,執行一個方法,並傳遞參數:

private void DoWork() {
Thread t = new Thread(new ParameterizedThreadStart(this.DoSomething));
t.Start("guozhijian");
}
private void DoSomething(object o) {
MessageBox.Show(o.ToString());

}

參數定義為object類型。

2. ThreadPool
眾所周知,新開一個線程代價是很高昂的,如果我們每個操作都新開一個線程,那麽太浪費了,於是,下面使用線程池。
無參數傳遞:

private void DoWork() {
ThreadPool.QueueUserWorkItem(new WaitCallback(this.DoSomething));
}
private void DoSomething(object o) {
MessageBox.Show("thread start");
}

有參數傳遞:

private void DoWork() {

ThreadPool.QueueUserWorkItem(new WaitCallback(this.DoSomething), "guozhijian");
}
private void DoSomething(object o) {
MessageBox.Show(o.ToString());
}

使用匿名方法更靈活:

private void DoWork() {
string name = "guozhijian";
ThreadPool.QueueUserWorkItem(new WaitCallback(delegate(object o){
MessageBox.Show(name);
}));
}

在匿名代碼段裏面可以直接訪問局部變量,不用在關心參數傳遞的問題

二. Invoke
1. this.Invoke
現在,在業務線程裏面執行完畢,要改變窗體控件的值了,此時,如果直接通過this得到控件的句柄,然後對它進行操作是會拋異常的,.Net WinForm Application裏面是不允許這樣的操作的。這是,可以調用Invoke方法

2.Invoke方法簽名:
object Control.Invoke(Delegate Method)
object Control.Invoke(Delegate Method, params object[] args)

3.使用自定義委托

private void DoWork() {
WaitCallback wc = new WaitCallback(this.DoSomething);
ThreadPool.QueueUserWorkItem(wc, "Guozhijian");
}
技術分享圖片
private delegate void MyInvokeDelegate(string name);

private void DoSomething(object o) {
this.Invoke(new MyInvokeDelegate(this.ChangeText), o.ToString());
}
技術分享圖片
private void ChangeText(string name) {
this.textBox1.Text = name;
}

4.使用System.Action:

private void DoWork() {
WaitCallback wc = new WaitCallback(this.DoSomething);
ThreadPool.QueueUserWorkItem(wc, "Guozhijian");
}
技術分享圖片
private void DoSomething(object o) {
this.Invoke(new Action<string>(this.ChangeText), o.ToString());
}
技術分享圖片
private void ChangeText(string name) {
this.textBox1.Text = name;
}

本例傳遞一個參數,System.Action有很多個重載,可以無參數(非泛型),而最多可以有四個參數,同樣采用匿名方法,不使用泛型形式的System.Action,如下:

private void DoWork() {
WaitCallback wc = new WaitCallback(this.DoSomething);
ThreadPool.QueueUserWorkItem(wc, "Guozhijian");
}
技術分享圖片
private void DoSomething(object o) {
this.Invoke(new Action(delegate() {
this.textBox1.Text = o.ToString();
}));
}

5.使用System.Func
如果Invoke調用主窗體操作之後,還希望在調用完得到一個返回值:

private void DoWork() {
WaitCallback wc = new WaitCallback(this.DoSomething);
ThreadPool.QueueUserWorkItem(wc, "Guozhijian");
}
技術分享圖片
private void DoSomething(object o) {
System.Func<string, int> f = new Func<string, int>(this.GetId);
object result = this.Invoke(f,o.ToString());
MessageBox.Show(result.ToString());
}
技術分享圖片
private int GetId(string name) {
this.textBox1.Text = name;
if (name == "Guozhijian") {
return 999;
}
else {
return 0;
}
}

result的值為 999。
System.Func同樣有很多泛形重載,這裏不贅述。

6.關於Invoke的擁有者:Control
本文例中都是用this來引用,這裏this替換為窗體任何一個控件的句柄都是OK的,因為Control.Invoke含義是將方法委托給擁有該Control的線程去執行。

【轉】淺述WinForm多線程編程與Control.Invoke的應用