C#異步調(diào)用的好處和方法分享
更新時間:2012年04月04日 18:35:35 作者:
我們要明確,為什么要進(jìn)行異步回調(diào)?眾所周知,普通方法運(yùn)行,是單線程的,如果中途有大型操作(如:讀取大文件,大批量操作數(shù)據(jù)庫,網(wǎng)絡(luò)傳輸?shù)龋紩?dǎo)致方法阻塞,表現(xiàn)在界面上就是,程序卡或者死掉,界面元素不動了,不響應(yīng)了
異步方法很好的解決了這些問題,異步執(zhí)行某個方法,程序立即開辟一個新線程去運(yùn)行你的方法,主線程包括界面就不會死掉了。異步如何開始,好理解,現(xiàn)在我們討論的是如何結(jié)束這個異步出來的新線程。
首先,異步出來的新線程,必須回收,不回收是浪費(fèi)資源的可恥行為,.NET也是不允許的,所以你別想鉆空子,俗話說,請神容易送神難,就是這個道理。下面你可以很容易想到,回收分為2種情況:主動回收和被動回收(當(dāng)然,這是我自己的理解,微軟可不是這么說的),主動回收就是,你去監(jiān)視那個線程,并且等待,當(dāng)異步方法完成了,就把異步線程回收,焦點(diǎn)回歸主線程,實(shí)際上就是上篇文章《C#異步初步》的那種情況,BeginInvoke之后又EndInvoke,如果在EndInvoke的時候,該異步線程沒有完成操作,那么整個程序,包括主線程,又在阻塞了,又會出現(xiàn)界面“死”的情況。要想解決這個問題,就使用“被動回收”方式,其中一個重要的辦法就是“異步回調(diào)”。 核心有二: A、 用回調(diào)函數(shù)(本例中為CallBackMethod),異步結(jié)束后,自動調(diào)用此回調(diào)函數(shù)。 B、 而不在主線程中手工等待異步結(jié)束,如上兩例中在主線程中調(diào)用EndInvoke。此種方法,是在回調(diào)函數(shù)中調(diào)用EndInvoke的。 異步回調(diào)的大概流程是這樣的:首先啟動異步,啟動參數(shù)加上異步結(jié)束時執(zhí)行的方法,然后這個異步線程就不用管了,最后當(dāng)這個異步線程自己完成工作了,就自動執(zhí)行啟動參數(shù)里的那個方法,這樣確實(shí)很省心,可是代碼寫起來,就很復(fù)雜了。 下面是搜藏的代碼:
//首先準(zhǔn)備好,要進(jìn)行異步的方法(能異步的,最好不多線程)
privatestringMethodName(intNum,outintNum2)
{
Num2=Num;
return"HelloWorld";
}
//程序終點(diǎn)
//異步完成時,執(zhí)行的方法(回調(diào)方法),此方法只能有IAsyncResult一個參數(shù),但是該參數(shù)幾乎萬能,可以傳遞object
privatevoidCallBackMethod(IAsyncResultar)
{
//從異步狀態(tài)ar.AsyncState中,獲取委托對象
DelegateNamedn=(DelegateName)ar.AsyncState;
//輸出參數(shù)
inti;
//一定要EndInvoke,否則你的下場很慘
stringr=dn.EndInvoke(outi,ar);
MessageBox.Show("異步完成嘍!i的值是"i.ToString()",r的值是"r);
}
//定義與方法同簽名的委托
privatedelegatestringDelegateName(intNum,outintNum2);
//程序入口
privatevoidRun()
{
//實(shí)例化委托并初賦值
DelegateNamedn=newDelegateName(MethodName);
//輸出參數(shù)
inti;
//實(shí)例化回調(diào)方法
//把AsyncCallback看成Delegate你就懂了,實(shí)際上AsyncCallback是一種特殊的Delegate,就像Event似的
AsyncCallbackacb=newAsyncCallback(CallBackMethod);
//異步開始
//如果參數(shù)acb換成null則表示沒有回調(diào)方法
//最后一個參數(shù)dn的地方,可以換成任意對象,該對象可以被回調(diào)方法從參數(shù)中獲取出來,寫成null也可以。參數(shù)dn相當(dāng)于該線程的ID,如果有多個異步線程,可以都是null,但是絕對不能一樣,不能是同一個object,否則異常
IAsyncResultiar=dn.BeginInvoke(1,outi,acb,dn);
//去做別的事
//…………
}
//最后的結(jié)果應(yīng)該是:i=1,r="HelloWorld"
//另外,如果可以,定義委托的時候可以選擇不用過多的修飾:
///<summary>
///定義委托
///</summary>
///<returns></returns>
publicdelegateboolAsyncdelegate();
///<summary>
///Callbackmethodmusthavethesamesignatureasthe
///AsyncCallbackdelegate
///</summary>
///<paramname="ar"></param>
privatevoidCallbackMethod(IAsyncResultar)
{
//Retrievethedelegate.
Asyncdelegatedlgt=(Asyncdelegate)ar.AsyncState;
//CallEndInvoketoretrievetheresults.
dlgt.EndInvoke(ar);
}
//其他方法中調(diào)用:
//異步執(zhí)行
//指定委托方法
Asyncdelegateisgt=newAsyncdelegate(icpInfo.Insert);
IAsyncResultar=isgt.BeginInvoke(newAsyncCallback(CallbackMethod),isgt);
首先,異步出來的新線程,必須回收,不回收是浪費(fèi)資源的可恥行為,.NET也是不允許的,所以你別想鉆空子,俗話說,請神容易送神難,就是這個道理。下面你可以很容易想到,回收分為2種情況:主動回收和被動回收(當(dāng)然,這是我自己的理解,微軟可不是這么說的),主動回收就是,你去監(jiān)視那個線程,并且等待,當(dāng)異步方法完成了,就把異步線程回收,焦點(diǎn)回歸主線程,實(shí)際上就是上篇文章《C#異步初步》的那種情況,BeginInvoke之后又EndInvoke,如果在EndInvoke的時候,該異步線程沒有完成操作,那么整個程序,包括主線程,又在阻塞了,又會出現(xiàn)界面“死”的情況。要想解決這個問題,就使用“被動回收”方式,其中一個重要的辦法就是“異步回調(diào)”。 核心有二: A、 用回調(diào)函數(shù)(本例中為CallBackMethod),異步結(jié)束后,自動調(diào)用此回調(diào)函數(shù)。 B、 而不在主線程中手工等待異步結(jié)束,如上兩例中在主線程中調(diào)用EndInvoke。此種方法,是在回調(diào)函數(shù)中調(diào)用EndInvoke的。 異步回調(diào)的大概流程是這樣的:首先啟動異步,啟動參數(shù)加上異步結(jié)束時執(zhí)行的方法,然后這個異步線程就不用管了,最后當(dāng)這個異步線程自己完成工作了,就自動執(zhí)行啟動參數(shù)里的那個方法,這樣確實(shí)很省心,可是代碼寫起來,就很復(fù)雜了。 下面是搜藏的代碼:
復(fù)制代碼 代碼如下:
//首先準(zhǔn)備好,要進(jìn)行異步的方法(能異步的,最好不多線程)
privatestringMethodName(intNum,outintNum2)
{
Num2=Num;
return"HelloWorld";
}
//程序終點(diǎn)
//異步完成時,執(zhí)行的方法(回調(diào)方法),此方法只能有IAsyncResult一個參數(shù),但是該參數(shù)幾乎萬能,可以傳遞object
privatevoidCallBackMethod(IAsyncResultar)
{
//從異步狀態(tài)ar.AsyncState中,獲取委托對象
DelegateNamedn=(DelegateName)ar.AsyncState;
//輸出參數(shù)
inti;
//一定要EndInvoke,否則你的下場很慘
stringr=dn.EndInvoke(outi,ar);
MessageBox.Show("異步完成嘍!i的值是"i.ToString()",r的值是"r);
}
//定義與方法同簽名的委托
privatedelegatestringDelegateName(intNum,outintNum2);
//程序入口
privatevoidRun()
{
//實(shí)例化委托并初賦值
DelegateNamedn=newDelegateName(MethodName);
//輸出參數(shù)
inti;
//實(shí)例化回調(diào)方法
//把AsyncCallback看成Delegate你就懂了,實(shí)際上AsyncCallback是一種特殊的Delegate,就像Event似的
AsyncCallbackacb=newAsyncCallback(CallBackMethod);
//異步開始
//如果參數(shù)acb換成null則表示沒有回調(diào)方法
//最后一個參數(shù)dn的地方,可以換成任意對象,該對象可以被回調(diào)方法從參數(shù)中獲取出來,寫成null也可以。參數(shù)dn相當(dāng)于該線程的ID,如果有多個異步線程,可以都是null,但是絕對不能一樣,不能是同一個object,否則異常
IAsyncResultiar=dn.BeginInvoke(1,outi,acb,dn);
//去做別的事
//…………
}
//最后的結(jié)果應(yīng)該是:i=1,r="HelloWorld"
//另外,如果可以,定義委托的時候可以選擇不用過多的修飾:
///<summary>
///定義委托
///</summary>
///<returns></returns>
publicdelegateboolAsyncdelegate();
///<summary>
///Callbackmethodmusthavethesamesignatureasthe
///AsyncCallbackdelegate
///</summary>
///<paramname="ar"></param>
privatevoidCallbackMethod(IAsyncResultar)
{
//Retrievethedelegate.
Asyncdelegatedlgt=(Asyncdelegate)ar.AsyncState;
//CallEndInvoketoretrievetheresults.
dlgt.EndInvoke(ar);
}
//其他方法中調(diào)用:
//異步執(zhí)行
//指定委托方法
Asyncdelegateisgt=newAsyncdelegate(icpInfo.Insert);
IAsyncResultar=isgt.BeginInvoke(newAsyncCallback(CallbackMethod),isgt);
相關(guān)文章
C# 關(guān)于AppDomain的一些總結(jié)
這篇文章主要介紹了C# 關(guān)于AppDomain的一些總結(jié),幫助大家更好的理解和使用c#,感興趣的朋友可以了解下2021-02-02C#實(shí)現(xiàn)簡單過濾非法字符實(shí)例
這篇文章主要介紹了C#實(shí)現(xiàn)簡單過濾非法字符的方法,涉及C#針對字符串遍歷與判斷的相關(guān)技巧,非常簡單實(shí)用,需要的朋友可以參考下2015-11-11winfrom 打印表格 字符串的封裝實(shí)現(xiàn)代碼 附源碼下載
以前寫打印都是根據(jù)打印機(jī)的型號,找開發(fā)類庫。然后在此基礎(chǔ)上開發(fā)。寫的多了自然就想到了封裝。這是還是想到了微軟,微軟封裝了PrintDocument的打印類。但這只是在低層對串口的封裝2013-02-02C#實(shí)現(xiàn)WinForm禁止最大化、最小化、雙擊標(biāo)題欄、雙擊圖標(biāo)等操作的方法
這篇文章主要介紹了C#實(shí)現(xiàn)WinForm禁止最大化、最小化、雙擊標(biāo)題欄、雙擊圖標(biāo)等操作的方法,涉及C#使用WinForm針對窗口操作的各種常用技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-08-08