|
最經(jīng)公司工作需要調(diào)用一個(gè)外部的webservice,同時(shí)要將傳出的數(shù)據(jù)進(jìn)行保存,以自己以前的習(xí)慣,就打算逐步操作,失敗啊,完全沒(méi)考慮過(guò)用戶體驗(yàn)效果,在同事指點(diǎn)下,意識(shí)到使用異步調(diào)用的好處,隨便將自己找的一些資料留以保存,以戒后誤! 我們要明確,為什么要進(jìn)行異步回調(diào)?眾所周知,普通方法運(yùn)行,是單線程的,如果中途有大型操作(如:讀取大文件,大批量操作數(shù)據(jù)庫(kù),網(wǎng)絡(luò)傳輸?shù)龋?,都?huì)導(dǎo)致方法阻塞,表現(xiàn)在界面上就是,程序卡或者死掉,界面元素不動(dòng)了,不響應(yīng)了。異步方法很好的解決了這些問(wèn)題,異步執(zhí)行某個(gè)方法,程序立即開(kāi)辟一個(gè)新線程去運(yùn)行你的方法,主線程包括界面就不會(huì)死掉了。異步如何開(kāi)始,好理解,現(xiàn)在我們討論的是如何結(jié)束這個(gè)異步出來(lái)的新線程。 首先,異步出來(lái)的新線程,必須回收,不回收是浪費(fèi)資源的可恥行為,.NET也是不允許的,所以你別想鉆空子,俗話說(shuō),請(qǐng)神容易送神難,就是這個(gè)道理。下面你可以很容易想到,回收分為2種情況:主動(dòng)回收和被動(dòng)回收(當(dāng)然,這是我自己的理解,微軟可不是這么說(shuō)的),主動(dòng)回收就是,你去監(jiān)視那個(gè)線程,并且等待,當(dāng)異步方法完成了,就把異步線程回收,焦點(diǎn)回歸主線程,實(shí)際上就是上篇文章《C#異步初步》的那種情況,BeginInvoke之后又EndInvoke,如果在EndInvoke的時(shí)候,該異步線程沒(méi)有完成操作,那么整個(gè)程序,包括主線程,又在阻塞了,又會(huì)出現(xiàn)界面“死”的情況。要想解決這個(gè)問(wèn)題,就使用“被動(dòng)回收”方式,其中一個(gè)重要的辦法就是“異步回調(diào)”。 核心有二: A、 用回調(diào)函數(shù)(本例中為CallBackMethod),異步結(jié)束后,自動(dòng)調(diào)用此回調(diào)函數(shù)。 B、 而不在主線程中手工等待異步結(jié)束,如上兩例中在主線程中調(diào)用EndInvoke。此種方法,是在回調(diào)函數(shù)中調(diào)用EndInvoke的。 異步回調(diào)的大概流程是這樣的:首先啟動(dòng)異步,啟動(dòng)參數(shù)加上異步結(jié)束時(shí)執(zhí)行的方法,然后這個(gè)異步線程就不用管了,最后當(dāng)這個(gè)異步線程自己完成工作了,就自動(dòng)執(zhí)行啟動(dòng)參數(shù)里的那個(gè)方法,這樣確實(shí)很省心,可是代碼寫起來(lái),就很復(fù)雜了。 下面是搜藏的代碼: //首先準(zhǔn)備好,要進(jìn)行異步的方法(能異步的,最好不多線程) private string MethodName(int Num, out int Num2) { Num2 = Num; return "HelloWorld"; } //程序終點(diǎn) //一定要EndInvoke,否則你的下場(chǎng)很慘 //定義與方法同簽名的委托 //程序入口 //最后的結(jié)果應(yīng)該是:i=1,r="HelloWorld" 另外,如果可以,定義委托的時(shí)候可以選擇不用過(guò)多的修飾: /// <summary> /// <summary> // Call EndInvoke to retrieve the results. //異步執(zhí)行 //指定委托方法 Asyncdelegate isgt = new Asyncdelegate(icpInfo.Insert); IAsyncResult ar = isgt.BeginInvoke(new AsyncCallback(CallbackMethod), isgt); |
|
|
來(lái)自: 命運(yùn)之輪 > 《C# and ASP.NET》