C#異步調用的好處和方法

 我們要明確,爲什麼要進行異步回調?衆所周知,普通方法運行,是單線程的,如果中途有大型操作(如:讀取大文件,大批量操作數據庫,網絡傳輸等),都會導致方法阻塞,表現在界面上就是,程序卡或者死掉,界面元素不動了,不響應了。異步方法很好的解決了這些問題,異步執行某個方法,程序立即開闢一個新線程去運行你的方法,主線程包括界面就不會死掉了。異步如何開始,好理解,現在我們討論的是如何結束這個異步出來的新線程。
       首先,異步出來的新線程,必須回收,不回收是浪費資源的可恥行爲,.NET也是不允許的,所以你別想鑽空子,俗話說,請神容易送神難,就是這個道理。下面你可以很容易想到,回收分爲2種情況:主動回收和被動回收(當然,這是我自己的理解,微軟可不是這麼說的),主動回收就是,你去監視那個線程,並且等待,當異步方法完成了,就把異步線程回收,焦點回歸主線程,實際上就是上篇文章《C#異步初步》的那種情況,BeginInvoke之後又EndInvoke,如果在EndInvoke的時候,該異步線程沒有完成操作,那麼整個程序,包括主線程,又在阻塞了,又會出現界面“死”的情況。要想解決這個問題,就使用“被動回收”方式,其中一個重要的辦法就是“異步回調”。
         
核心有二:
    
A、   用回調函數(本例中爲CallBackMethod
),異步結束後,自動調用此回調函數。    
    
B、   
而不在主線程中手工等待異步結束,如上兩例中在主線程中調用EndInvoke。此種方法,是在回調函數中調用EndInvoke的。
     異步回調的大概流程是這樣的:首先啓動異步,啓動參數加上異步結束時執行的方法,然後這個異步線程就不用管了,最後當這個異步線程自己完成工作了,就自動執行啓動參數裏的那個方法
,這樣確實很省心,可是代碼寫起來,就很複雜了。
    下面是蒐藏的代碼:

 //
首先準備好,要進行異步的方法(能異步的,最好不多線程)
private string MethodName(int Num, out int Num2)
{
               Num2 = Num;
               return "HelloWorld";
}

//程序終點
//異步完成時,執行的方法(回調方法),此方法只能有IAsyncResult一個參數,但是該參數幾乎萬能,可以傳遞object
private void CallBackMethod(IAsyncResult ar)
{
               //從異步狀態ar.AsyncState中,獲取委託對象
               DelegateName dn = (DelegateName)ar.AsyncState;
               //輸出參數
               int i;

               //一定要EndInvoke,否則你的下場很慘
               string r = dn.EndInvoke(out i, ar);
               MessageBox.Show("異步完成嘍!i的值是" i.ToString() ",r的值是" r);
}

//定義與方法同簽名的委託
private delegate string DelegateName(int Num, out int Num2);

//程序入口
private void Run()
{
               //實例化委託並初賦值
               DelegateName dn = new DelegateName(MethodName);
               //輸出參數
               int i;
               //實例化回調方法
               //把AsyncCallback看成Delegate你就懂了,實際上AsyncCallback是一種特殊的Delegate,就像Event似的
               AsyncCallback acb = new AsyncCallback(CallBackMethod);
               //異步開始
               //如果參數acb換成null則表示沒有回調方法
               //最後一個參數dn的地方,可以換成任意對象,該對象可以被回調方法從參數中獲取出來,寫成null也可以。參數dn相當於該線程的ID,如果有多個異步線程,可以都是null,但是絕對不能一樣,不能是同一個object,否則異常
               IAsyncResult iar = dn.BeginInvoke(1, out i, acb, dn);
               //去做別的事
               //…………
}

//最後的結果應該是:i=1,r="HelloWorld"


另外,如果可以,定義委託的時候可以選擇不用過多的修飾:

        /// <summary>
        /// 定義委託
        /// </summary>
        /// <returns></returns>
        public delegate bool Asyncdelegate();

        /// <summary>
        /// Callback method must have the same signature as the
        /// AsyncCallback delegate
        /// </summary>
        /// <param name="ar"></param>
        private void CallbackMethod(IAsyncResult ar)
        {
            // Retrieve the delegate.
            Asyncdelegate dlgt = (Asyncdelegate)ar.AsyncState;

            // Call EndInvoke to retrieve the results.
            dlgt.EndInvoke(ar);
        }

  其他方法中調用:
        //異步執行      
        //指定委託方法      
        Asyncdelegate isgt = new Asyncdelegate(icpInfo.Insert);
        IAsyncResult ar = isgt.BeginInvoke(new AsyncCallback(CallbackMethod), isgt);
        
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章