C#中的異步調用例子

C#異步調用的應用實踐最經公司工作需要調用一個外部的webservice,同時要將傳出的數據進行保存,以自己以前的習慣,就打算逐步操作,失敗啊,完全沒考慮過用戶體驗效果,在同事指點下,意識到使用C#異步調用的好處,隨便將自己找的一些資料留以保存,以戒後誤!

我們要明確,爲什麼要進行C#異步調用回調?衆所周知,普通方法運行,是單線程的,如果中途有大型操作(如:讀取大文件,大批量操作數據庫,網絡傳輸等),都會導致方法阻塞,表現在界面上就是,程序卡或者死掉,界面元素不動了,不響應了。C#異步調用方法很好的解決了這些問題,C#異步調用執行某個方法,程序立即開闢一個新線程去運行你的方法,主線程包括界面就不會死掉了。C#異步調用如何開始,好理解,現在我們討論的是如何結束這個C#異步調用出來的新線程。

首先,C#異步調用出來的新線程,必須回收,不回收是浪費資源的可恥行爲,.NET也是不允許的,所以你別想鑽空子,俗話說,請神容易送神難,就是這個道理。下面你可以很容易想到,回收分爲2種情況:主動回收和被動回收(當然,這是我自己的理解,微軟可不是這麼說的),主動回收就是,你去監視那個線程,並且等待,當C#異步調用方法完成了,就把C#異步調用線程回收,焦點回歸主線程,實際上就是上篇文章《C#C#異步調用初步》的那種情況,BeginInvoke之後又EndInvoke,如果在EndInvoke的時候,該C#異步調用線程沒有完成操作,那麼整個程序,包括主線程,又在阻塞了,又會出現界面“死”的情況。要想解決這個問題,就使用“被動回收”方式,其中一個重要的辦法就是“C#異步調用回調”。

C#異步調用核心有二:

A、 用回調函數(本例中爲CallBackMethod),C#異步調用結束後,自動調用此回調函數。

B、 而不在主線程中手工等待C#異步調用結束,如上兩例中在主線程中調用EndInvoke。此種方法,是在回調函數中調用EndInvoke的。

C#異步調用回調的大概流程是這樣的:首先啓動C#異步調用,啓動參數加上C#異步調用結束時執行的方法,然後這個C#異步調用線程就不用管了,最後當這個C#異步調用線程自己完成工作了,就自動執行啓動參數裏的那個方法,這樣確實很省心,可是代碼寫起來,就很複雜了。

下面是蒐藏的代碼:

  1. //首先準備好,要進行C#異步調用的方法(能C#異步調用的,最好不多線程)  
  2. private string MethodName(int Num, out int Num2)  
  3. {  
  4.  Num2 = Num;  
  5.  return "HelloWorld";  
  6. }   
  7.  
  8. //程序終點  
  9. //C#異步調用完成時,執行的方法(回調方法),  
  10. //此方法只能有IAsyncResult一個參數,但是該參數幾乎萬能,可以傳遞object  
  11. private void CallBackMethod(IAsyncResult ar)  
  12. {  
  13.  //從C#異步調用狀態ar.AsyncState中,獲取委託對象  
  14.  DelegateName dn = (DelegateName)ar.AsyncState;  
  15.  //輸出參數  
  16.  int i;   
  17.  
  18.  //一定要EndInvoke,否則你的下場很慘  
  19.  string r = dn.EndInvoke(out i, ar);  
  20.  MessageBox.Show("C#異步調用完成嘍!i的值是" i.ToString() ",r的值是" r);  
  21. }   
  22.  
  23. //定義與方法同簽名的委託  
  24. private delegate string DelegateName(int Num, out int Num2);   
  25.  
  26. //程序入口  
  27. private void Run()  
  28. {  
  29.  //實例化委託並初賦值  
  30.  DelegateName dn = new DelegateName(MethodName);  
  31.  //輸出參數  
  32.  int i;  
  33.  //實例化回調方法  
  34.  //把AsyncCallback看成Delegate你就懂了,  
  35. //實際上AsyncCallback是一種特殊的Delegate,就像Event似的  
  36.  AsyncCallback acb = new AsyncCallback(CallBackMethod);  
  37.  //C#異步調用開始  
  38.  //如果參數acb換成null則表示沒有回調方法  
  39.  //最後一個參數dn的地方,可以換成任意對象,  
  40. //該對象可以被回調方法從參數中獲取出來,寫成null也可以。  
  41. //參數dn相當於該線程的ID,如果有多個C#異步調用線程,  
  42. //可以都是null,但是絕對不能一樣,不能是同一個object,否則異常  
  43.  IAsyncResult iar = dn.BeginInvoke(1, out i, acb, dn);  
  44.  //去做別的事  
  45.  //…………  
  46. }   
  47.  
  48. //最後的結果應該是:i=1,r="HelloWorld"  

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

  1. /// ﹤summary﹥  
  2. /// 定義委託  
  3. /// ﹤/summary﹥  
  4. /// ﹤returns﹥﹤/returns﹥  
  5. public delegate bool Asyncdelegate();   
  6.  
  7. /// ﹤summary﹥  
  8. /// Callback method must have the same signature as the  
  9. /// AsyncCallback delegate  
  10. /// ﹤/summary﹥  
  11. /// ﹤param name="ar"﹥﹤/param﹥  
  12. private void CallbackMethod(IAsyncResult ar)  
  13. {  
  14. // Retrieve the delegate.  
  15. Asyncdelegate dlgt = (Asyncdelegate)ar.AsyncState;   
  16.  
  17. // Call EndInvoke to retrieve the results.  
  18. dlgt.EndInvoke(ar);  
  19. }  

其他方法中調用:

  1. //C#異步調用執行  
  2. //指定委託方法  
  3. Asyncdelegate isgt = new Asyncdelegate(icpInfo.Insert);  
  4. IAsyncResult ar = isgt.BeginInvoke(  
  5. new AsyncCallback(CallbackMethod), isgt); 

我的完整例子,保存。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;

namespace ConsoleApplication1
{
    public delegate void MyEventDelegate(int value);


    public class Publisher
    {
        public MyEventDelegate myEvent;
    }

    public class Subscriber {

        public void myMethod(int value)
        {
            Console.WriteLine(value);
        }  
    }

    class Program
    {
        static void Main(string[] args)
        {
            Publisher p = new Publisher();

            Subscriber s1 = new Subscriber();
            Subscriber s2 = new Subscriber();

           // p.myEvent = new MyEventDelegate(s1.myMethod);
            p.myEvent = s1.myMethod;
           // p.myEvent += s2.myMethod;

            //p.myEvent(100);
            //p.myEvent.Invoke(100);
            AsyncCallback acb = new AsyncCallback(CallBackMethod); 
           
            p.myEvent.BeginInvoke(100, acb, p.myEvent);
            Console.Read();

        }

private static void CallBackMethod(IAsyncResult ar) 

 //從C#異步調用狀態ar.AsyncState中,獲取委託對象 
 MyEventDelegate dn = (MyEventDelegate)ar.AsyncState;

 //Console.WriteLine("C#異步調用完成嘍!i的值是{0}", ar.AsyncState.ToString()); 

 //一定要EndInvoke,否則你的下場很慘 
 dn.EndInvoke(ar);
 Console.WriteLine("C#異步調用完成嘍!i的值是{0}",ar.AsyncState.ToString()); 
}  
      
    }
}

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章