c#中Invoke與BeginInvoke的用法及說明
c# Invoke與BeginInvoke
最近在學(xué)習(xí)線程時,發(fā)現(xiàn)當(dāng)我創(chuàng)建的線程需要訪問UI界面的時,會發(fā)生異常,原因是我在跨線程調(diào)用主線程的控件,因此windows GUI編程有一個規(guī)則,就是只能通過創(chuàng)建控件的線程來操作控件的數(shù)據(jù),否則就可能產(chǎn)生不可預(yù)料的結(jié)果。
有時候,我們不得不跨線程調(diào)用主界面的控件來進(jìn)行操作,所以為了方便的解決問題,.net為我們提供了Invoke 與beginInvoke
Invoke 與begininvoke區(qū)別在于,invoke會阻塞當(dāng)前線程,直到invoke調(diào)用結(jié)束,才會繼續(xù)執(zhí)行下去,而begininvoke 則可以異步進(jìn)行調(diào)用,也就是該方法封送完畢后馬上返回,不會等待委托方法的執(zhí)行結(jié)束,調(diào)用者線程將不會被阻塞。但是調(diào)用者也可以使用EndInvoke方法或者其它類似WaitHandle機制等待異步操作的完成。
先講下Invoke
? ? ? ? // 定義委托函數(shù) ,委托函數(shù)與被委托函數(shù)必須要有相同返回值和參數(shù)列表 ? ? ? ? public delegate void myDelegate(string str); ? ? ? ? ? public void _invoke_myDelegate(String str) ? ? ? ? { ? ? ? ? ? ? // invokeRequired 獲取一個bool值 判斷調(diào)用控件是否必須要調(diào)用invoke方法 ? ? ? ? ? ? // 如果調(diào)用對象在其他線程,則返回true,否則返回false ? ? ? ? ? ? if (this.InvokeRequired) ? ? ? ? ? ? { ? ? ? ? ? ? ? ?/* Action<string> action = new Action<string>(_invoke_myDelegate);*/ ? ? ? ? ? ? ? ?// 確定調(diào)用對象在其他線程 則調(diào)用invoke函數(shù) 它會返回到擁有這個控件的線程上 ? ? ? ? ? ? ? ?// ?利用委托函數(shù),再次調(diào)用被委托函數(shù),str為委托函數(shù)的參數(shù)列表 ? ? ? ? ? ? ? ?this.Invoke(new myDelegate(_invoke_myDelegate), str); ? ? ? ? ? ? } ? ? ? ? ? ? // 當(dāng)委托函數(shù)執(zhí)行時, 此時已經(jīng)回到控件線程,可以直接調(diào)用控件label ? ? ? ? ? ? label1.Text = str; ? ? ? ? }
這里Invoke 必須等委托函數(shù)調(diào)用完成之后,才會執(zhí)行后面操作,那么當(dāng)我們的委托函數(shù)執(zhí)行的是一個非常耗時的操作
這樣線程就會被阻塞,造成用戶界面卡頓的情況,所以,為了解決invoke同步的問題,還有一種就是beginInvoke
BeginInvoke
BeginInvoke方法觸發(fā)你的異步方法,它和你想要執(zhí)行的異步方法有相同的參數(shù)。
另外還有兩個可選參數(shù)
- 1.第一個是AsyncCallback委托是異步完成的回調(diào)方法。
- 2.第二個是用戶自定義對象,該對象將傳遞到回調(diào)方法中。
BeginInvoke立即返回并且不等待完成異步的調(diào)用(繼續(xù)執(zhí)行該下面的代碼,不需要等待)。
BeginInvoke返回IAsyncResult接口,可用于檢測異步調(diào)用的過程。
通過EndInvoke方法檢測異步調(diào)用的結(jié)果。如果異步調(diào)用尚未完成,EndInvoke將阻塞調(diào)用線程,直到它完成。EndInvoke參數(shù)包括out和ref參數(shù)。
不管怎么么樣,調(diào)用了beginInvoke ,就必須調(diào)用endInvoke 結(jié)束異步,
那我們怎么才能知道什么時候異步結(jié)束呢?
常見四種方法:
- 1.做一些其他操作,然后調(diào)用EndInvoke方法阻塞線程直到該方法完成。
- 2.使用IAsyncResult.AsyncWaitHandle屬性,使用它的WaitOne方法阻塞線程直到收到WaitHandle信號,然后調(diào)用EndInvoke。
- 3.檢查BeginInvoke返回值IAsyncResult的狀態(tài)來決定方法是否完成,然后調(diào)用EndInvoke方法。
- 4.通過在BeginInvoke方法中傳遞該委托,在回調(diào)方法中調(diào)用該委托的EndInvoke方法。
? ?AsyncMethodCaller caller = new AsyncMethodCaller(TestMethodAsync); // caller 為委托函數(shù) ? ? ? ? ? ? int threadid = 0; ? ? ? ? ? ? //開啟異步操作 ? ? ? ? ? ? IAsyncResult result = caller.BeginInvoke(1000, out threadid, null, null); ? ? ? ? ? ? for (int i = 0; i < 10; i++) ? ? ? ? ? ? { ? ? ? ? ? ? ? ? Console.WriteLine("其它業(yè)務(wù)" + i.ToString()); ? ? ? ? ? ? } ? ? ? ? ? ? //調(diào)用EndInvoke,等待異步執(zhí)行完成 ? ? ? ? ? ? Console.WriteLine("等待異步方法TestMethodAsync執(zhí)行完成"); ? ? ? ? ? ? //等待異步執(zhí)行完畢信號 ? ? ? ? ? ? //result.AsyncWaitHandle.WaitOne(); ? ? ? ? ? ? //Console.WriteLine("收到WaitHandle信號"); ? ? ? ? ? ? //通過循環(huán)不停的檢查異步運行狀態(tài) ? ? ? ? ? ? while (result.IsCompleted==false) ? ? ? ? ? ? { ? ? ? ? ? ? ? ? Thread.Sleep(100); ? ? ? ? ? ? ? ? Console.WriteLine("異步方法,running........"); ? ? ? ? ? ? } ? ? ? ? ? ? //異步結(jié)束,拿到運行結(jié)果 ? ? ? ? ? ? string res = caller.EndInvoke(out threadid, result); ? ? ? ? ? ? //顯示關(guān)閉句柄 ? ? ? ? ? ? result.AsyncWaitHandle.Close(); ? ? ? ? ? ? Console.WriteLine("關(guān)閉了WaitHandle句柄"); static string TestMethodAsync(int callDuration, out int threadId) ? ? ? ? { ? ? ? ? ? ? Stopwatch sw = new Stopwatch(); ? ? ? ? ? ? sw.Start(); ? ? ? ? ? ? Console.WriteLine("異步TestMethodAsync開始"); ? ? ? ? ? ? for (int i = 0; i < 5; i++) ? ? ? ? ? ? { ? // 模擬耗時操作 ? ? ? ? ? ? ? ? Thread.Sleep(callDuration); ? ? ? ? ? ? ? ? Console.WriteLine("TestMethodAsync:" + i.ToString()); ? ? ? ? ? ? } ? ? ? ? ? ? sw.Stop(); ? ? ? ? ? ? threadId = Thread.CurrentThread.ManagedThreadId; ? ? ? ? ? ? return string.Format("耗時{0}ms.", sw.ElapsedMilliseconds.ToString()); ? ? ? ? }
總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
C#采用FileSystemWatcher實現(xiàn)監(jiān)視磁盤文件變更的方法
這篇文章主要介紹了C#采用FileSystemWatcher實現(xiàn)監(jiān)視磁盤文件變更的方法,詳細(xì)分析了FileSystemWatcher的用法,并以此為基礎(chǔ)實現(xiàn)監(jiān)視磁盤文件變更,是非常實用的技巧,具有一定的借鑒價值,需要的朋友可以參考下2014-11-11C# Dynamic之:ExpandoObject,DynamicObject,DynamicMetaOb的應(yīng)用(下)
本篇文章是對C#中ExpandoObject,DynamicObject,DynamicMetaOb的應(yīng)用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05C#中使用ADOMD.NET查詢多維數(shù)據(jù)集的實現(xiàn)方法
這篇文章主要介紹了C#中使用ADOMD.NET查詢多維數(shù)據(jù)集的實現(xiàn)方法,詳細(xì)講述了C#中使用ADOMD.NET查詢多維數(shù)據(jù)集的原理與實現(xiàn)技巧,需要的朋友可以參考下2014-10-10