我們在寫Remoting程式或者其他的一些應用程式的時候難免要和線程打交道,.Net使我們很容易就可以建立一個線程,但是它提供的建立線程和啟動線程的方法沒有明顯的提供參數,假如我們要用線程來啟動類裡面一個帶參數的方法該怎麼辦?下面就簡單的介紹如何使用.NET提供的豐富的架構來實作這個功能。為了可以生動詳細的介紹整個過程,我建立下面的一個.NET類,它也是要用線程啟動的方法的載體。類如下所示:
using System;
namespace WindowsApplication1
{
///
/// Summary description for UrlFetcher.
///
public class MyClass{
// for method 1
private string _parameter;
public MyClass(string parameter){
this._parameter = parameter;
}
public void MyMethod1(){
if(this._parameter!=null){
// do something
Console.Write(this._parameter);
}
}
// for method 2
public MyClass(){}
// this method is private,But it can be public or other
private void MyMethod2(string parameter){
// do something
Console.Write(parameter);
}
// Because delegate WaitCallback's parameter Type is object
// I will convert it to string.
public void MyMethod2(object parameter){
this.MyMethod2((string)parameter);
}
// for method 3
public string MyMethod3(string parameter){
return "參數值為:"+parameter;
}
// for mutil-parameters passed
public string MyMutilParameters(string param1,string param2){
return "參數1和參數2連接配接結果為:"+param1+param2;
}
}
}
|
嘿嘿,我的英語不行,注釋寫的不好請見諒(因為使用的是英文的),希望沒有影響您的閱讀。我想我有必要簡單的說一下上面這個類裡裡面的所包含的内容。首先包含兩個構造函數,一個帶參數一個不帶(這裡可是有意安排的)。通過類中其他方法的名字我想您一定猜出來我将介紹3種方法來傳遞參數,接下來我将逐一介紹。首先我們看看如何啟動一個線程,首先我們可以用一個函數來執行個體化ThreadStart委托的一個執行個體,然後在使用這個執行個體作為參數new線程(Thread)對象,最後将這個線程Start就可以了,想了解更多請參考MSDN文檔的Thread部分。
為了測試我們的結果我建立了一個WinForm的工程,其中有一個Form和4個按鈕,如果你需要所有的源碼請發送郵件到[email protected],如果我有時間我會給你發過去的。接下來是每種方法的較長的描述。
1、使用構造函數來傳遞參數
衆所周知,我們可以使用一個帶參數的構造函數來構造對象,既然這樣我們可以利用構造函數先将要使用的參數值傳遞到對象裡面的内部變量上,然後再使用一個無參數的方法來使用這個參數(假裝參數)。簡單的說就是,在類裡面聲明一個變量專門用來儲存函數需要的參數,函數變成無參的形式。這種方法的最大問題就是破壞了封裝性,雖然我們不能直接方法這些變量但是隐患總是存在的(或者說看上去不爽都可以)。下面代碼片斷給出了如何使用這個方法來傳遞參數的詳細内容,這也是上面提到的4個按鈕中的一個按鈕(Button1)的Click代碼。為了有參數可傳我在WinForm全局定義了一個如下的變量:
// This is parameter's value
private string myParameter = "ParameterValue/n";
|
按鈕事件如下所示:
// passed parameters to thread by construct
private void button1_Click(object sender, System.EventArgs e) {
MyClass instance = new MyClass(myParameter);
new Thread (new ThreadStart(instance.MyMethod1)).Start();
}
|
正如上面所說的,我們使用構造函數傳遞參數到類裡面去,然後在使用上面所說的方法啟動了一個線程,我們可以在運作該程式後的output視窗中看到MyMethod1的執行結果是(你也可以用一個TextBox或者其他什麼東西直接顯示在WinForm上):ParameterValue 。看看函數體就知道這個結果是正确的。是不是很簡單。
2、使用ThreadPool來實作參數的傳遞
我們首先可以看看MSDN對ThreadPool是怎麼形容的,Provides a pool of threads that can be used to post work items, process asynchronous I/O, wait on behalf of other threads, and process timers.檢視它的方法集合其中有一個叫:QueueUserWorkItem 的方法,該類以及該方法的詳細資訊請參考MSDN相關幫助。這裡需要注意的就是QueueUserWorkItem方法的參數,參數WaitCallback是一個委托類型,第二個參數就是該委托執行個體(用函數執行個體化以後,也就是一個函數)所需要的參數,是object類型的。詳細内容請看下面的代碼。
// passed parameter to thread by ThreadPool
private void button2_Click(object sender, System.EventArgs e) {
MyClass instance = new MyClass();
ThreadPool.QueueUserWorkItem (new WaitCallback (instance.MyMethod2),myParameter);
}
|
因為QueueUserWorkItem的二個參數是object類型的是以我們要在MyClass裡面定義兩個MyMethod2的重裁版本,目的就是為了滿足該方法的參數。同樣我們将參數myParameter傳遞進去了,運作程式,當我們點選Button2的時候在output視窗中将會出現MyMethod2将myParameter作為參數執行的結果顯示出來。
3、接下來是最後一種方法使用異步委托來實作參數的傳遞
同樣,有關委托的詳細資訊可以參考MSDN,上面說的非常詳細。我們這裡要使用到BeginInvoke和EndInvoke方法。首先我們給出傳遞一個參數的方法如下所示:
// passed parameter by asynchronous delegate
delegate string MyMethod3Delegate(string parameter);
private void button3_Click(object sender, System.EventArgs e) {
MyClass instance = new MyClass();
MyMethod3Delegate myMethod3 = new MyMethod3Delegate(instance.MyMethod3);
myMethod3.BeginInvoke("parameterValue",new AsyncCallback(AfterMyMothod3),null);
}
public void AfterMyMothod3(IAsyncResult result){
AsyncResult async = (AsyncResult) result;
MyMethod3Delegate DelegateInstance = (MyMethod3Delegate) async.AsyncDelegate;
Console.WriteLine ("函數調用傳回值:{0}/n", DelegateInstance.EndInvoke(result));
}
|
首先為了使用委托我們聲明了一個MyMethod3Delegate的委托,該委托說明一個參數和傳回值為string的函數是符合條件的,是以我們在MyClass裡面定義了一個MyMethod3的方法。該函數的型構符合上面的委托,是以我們可以在Button3點選的時候用這個方法執行個體化一個委托,然後我們使用異步的方式調用這個方法,為了得到傳回結果我們寫了AfterMyMothod3方法用來顯示該函數的執行結果。運作程式點選Button3可以看到Output中輸出的結果為MyMethod3帶參數執行的結果。最後我給出如何傳遞多個參數的方法,我的例子是傳遞2個參數。代碼如下:
// mutil-parameters passed
delegate string MyMutilParamsDelegate(string parameter1,string parameter2);
private void button4_Click(object sender, System.EventArgs e) {
MyClass instance = new MyClass();
MyMutilParamsDelegate mutilParams = new MyMutilParamsDelegate(instance.MyMutilParameters);
mutilParams.BeginInvoke("param1","params2",new AsyncCallback(AfterMutilParams),null);
}
public void AfterMutilParams(IAsyncResult result){
AsyncResult async = (AsyncResult) result;
MyMutilParamsDelegate DelegateInstance = (MyMutilParamsDelegate) async.AsyncDelegate;
Console.WriteLine ("多參數函數調用傳回結果:{0}/n", DelegateInstance.EndInvoke(result));
}
|