上一篇文章(什么是.Net的异步机制(委托Delegate) - step 1),我已经解释了什么是异步编程,那么现在我们就开始具体的说怎样异步编程.

 

我们怎样进行异步编程/开发?

 

现在扩充下上篇文章的类(AsyncTest),提供更多的例子并从中做下简单的对比, 从新的认识下异步的内部机制,下面我们增加一个新的委托

1,我们添加一个新方法(计算年薪YearlySalary)

public decimal YearlySalary(decimal salary, int monthCount, decimal bonus);

2,为这个方法增加异步的功能,这样我们仍然使用委托(Delegate)

public delegate decimal SalaryEventHandler(decimal salary, int monthCount, decimal bonus);

 

经过简单修改后,下面是我们新的AsyncTest

Code1

 1什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2//我们使用委托来提供.Net的异步机制
 2什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2public delegate string AsyncEventHandler(string name); // 对应Hello 方法
 3什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2public delegate decimal SalaryEventHandler(decimal salary, int monthCount, decimal bonus); // 对应YearlySalary方法
 4什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2public class AsyncTest
 5}

 

这里用.NET Reflector 5 来反编译,之所以用这个,因为比微软的会更加清晰明了.如果想了解这个工具的朋友可查看(http://reflector.red-gate.com/)

什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2

1

开始我先对图1中的小图标进行个简单的解释

什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2 = (Class)   什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2 = 类继承的基类 什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2 = sealed(委托)

什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2 = 类的构造函数 什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2 = 方法 什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2virtual方法

 

下面我们先比较下SalaryEventHandler AsyncEventHandler委托的异同.

1)      SalaryEventHandler

public delegate decimal SalaryEventHandler(decimal salary, int monthCount, decimal bonus);

什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2

2.1

编译器生成的类Code2.1(2.1)
Code 2.1

 1什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2    public sealed class SalaryEventHandler : MulticastDelegate
 2 bonus,
 AsyncCallback callback, 
object @object)
 6    }

2)      AsyncEventHandler

public delegate string AsyncEventHandler(string name);

什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2

2.2

编译器生成的类Code2.2(2.2)

Code2.2

 1什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2    public sealed class AsyncEventHandler : MulticastDelegate
 2    }

 

对比两个委托(事实上是一个sealed 的类),都继承于System.MuliticaseDelegate, 三个virtual Invoke / BeginInvoke / EndInvoke 方法.

//同步方法

Invoke : 参数的个数,类型, 返回值都不相同

 

//异步方法,作为一组来说明

BeginInvoke : 参数的个数和类型不同,返回值相同

EndInvoke : 参数相同,返回值不同

 

这里我们先介绍下 Invoke这个方法, 我们用SalaryEventHandler委托为例(直接调用Code 1 的类)

Code 3

 1什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2class Program
 2}

输出的结果

什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2

3

从结果可以看出,他们是同一个线程调用的(都是#10).这就说明[1],[2],[3]是同步调用

[2],[3]对比[1], 只不过[2],[3]是通过委托的方式(其实我们可以说成“通过代理的方式完成”),[1]是直接的调用.举一个我们平常生活中例子:买机票,我们到代理点购买机票而不是直接跑到机场购买,就好像我们叫别人帮我们买机票一样,最后到手的机票是一样的, SalaryEventHandler就是我们的代理点.所以用代理的方式还是直接调用的方式,他们提供的参数和返回值必须是一样的.

 

接下来我们开始讲异步机制核心的两个方法BeginInvoke/EndInvoke,他们作为一个整体来完成Invoke方法的调用,不同于Inoke方法的是他们是异步执行(另外开一个线程执行),下面先解释下他们的作用

BeginInvoke : 开始一个异步的请求,调用线程池中一个线程来执行
EndInvoke : 完成异步的调用, 处理返回值 异常错误.

注意: BeginInvokeEndInvoke必须成对调用.即使不需要返回值,但EndInvoke还是必须调用,否则可能会造成内存泄漏.


我们来对比下 SalaryEventHandler AsyncEventHandler委托反编译BeginInoke后的异同.

SalaryEventHandler 委托:

public virtual IAsyncResult BeginInvoke(decimal salary, int monthCount, decimal bonus, AsyncCallback callback, object @object)

AsyncEventHandler 委托:

public virtual IAsyncResult BeginInvoke(string name, AsyncCallback callback, object @object)

 

可以看出参数的个数和类型是不同的,我们把焦点放到他们的相同点上,

1,返回值是相同的: 返回IAsyncResult 对象(异步的核心). IAsyncResult是什么呢? 简单的说,存储异步操作的状态信息的一个接口,也可以用他来结束当前异步.具体的可以看下 http://msdn.microsoft.com/zh-cn/library/system.iasyncresult(VS.80).aspx

 

2,编译器会根据委托的参数个数和类型生成相应的BeginInvoke方法,只有最后两个参数是永远相同的,他提供一个AsyncCallback 委托(public delegate void AsyncCallback(IAsyncResult ar);) 和一个 Object 对象.

 

我们再来看看EndInvoke的异同.

SalaryEventHandler 委托:

public virtual decimal EndInvoke(IAsyncResult result)

AsyncEventHandler 委托:

public virtual string EndInvoke(IAsyncResult result)

 

EndInvoke的参数是一样的, 唯一是在是返回值不同(他们会根据自己委托的返回值类型生成自己的类型)

 

,下面我会通过例子来说明BeginInvoke/EndInvoke,还是使用SalaryEventHandler委托为例(直接调用Code 1 的类)

 

.Net Framework 提供了两种方式来使用异步方法

第一种: 通过IAsyncResult 对象

Code 4.1

 1什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2class Program
 2}

第二种: 通过回调函数. 使用倒数第二个参数AsyncCallback 委托(public delegate void AsyncCallback(IAsyncResult ar);) ,建议使用这种方法.

Code 4.2

 1什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2class Program
 2}

 

BeginInvoke最后一个参数是做什么的呢?我把Code 4.2 方法修改下.

Code 4.3

 1什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2class Program
 2}

 

异步的异常处理

接下来再讲讲EndInvoke,获取最后的返回值之外,他的一个重要的应用在引发异常来从异步操作返回异常

Code 5

 1什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2class Program
 2}

 

我们主动在得到原始的异常.

 

说到这里,大家是否可以简单的应用委托来开始自己的异步操作呢? 下面看看我是怎样为我自己的类添加异步的.

1, 类的定义,需要遵循.Net Framework 的规则

1)同步和异步是同时并存的

2)从最上面的两个委托SalaryEventHandler AsyncEventHandler生成的BeginInvoke / EndInvoke 对比中看出,我们也来定义我们自己的异步方法,我们遵循微软设计师异步方法设计的规则,Begin+同步方法名 / End+同步方法名

BeginXXX 必须返回IAsyncResult对象,后两位参数必须为AsyncCallback callback, object state,前面的参数和同步方法的参数一样

EndXXX 参数必须为IAsyncResult对象,返回值为同步方法的返回值

 

Code 6.1

 1什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2public class AsyncTest
 2}

 

2: 调用我们编写的类

Code 6.2

 1什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2class Program
 2}

 

:Hello的方法的异步重构在下面的代码中可以下载到.

 

下一篇中我们会说说异步中的一些高级应用,异步的核心,还有微软.Net Framework 为我们提供的各种类中(具有异步方法的类),我们是怎么使用这些方法的.最后要提醒下大家:滥用异步,会影响性能,而且增加编程难度什么是.Net的异步机制(Invoke,BeginInvoke,EndInvoke) - step 2


以上有word 文档直接粘贴,排版可能不太看,你可以通过下面来下载相应的代码/文档

1,文档

2,代码 (VS2008开发,.Net Framework 2.0(C Sharp)编写)

文章为原创,如果需要引用,请保留原地址. 有什么问题/错误的地方请联系 fox7805034 (at) hotmail.com

相关文章:

  • 2021-07-24
  • 2022-01-31
  • 2021-09-19
猜你喜欢
  • 2022-12-23
  • 2021-09-18
  • 2021-11-24
相关资源
相似解决方案