C#中的异步调用例子

时间:2022-08-27 13:37:44

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()); 
}  
      
    }
}