.Net的异步机制(委托Delegate) - STEP 1
在阅读下面知识前,我已经认为你已经具有c#的基础,包括简单的委托知识; 代码使用VS2008开发,但是会在.Net Framework 2.0(C Sharp)编写
什么是.Net异步机制呢?
在解释这个话题前,我们先看看同步的程序,就是我们常用的Hello World 程序。
Code 1:
class Program2
{3
static void Main(string[] args)4
{5
// 查看当前的线程ID, 是否线程池里面的线程6
Console.WriteLine("1,Thread ID:#{0},Is PoolThread?{1}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread);
7

8
AsyncTest test = new AsyncTest();9
string val = test.Hello("Andy Huang");10

11
Console.WriteLine(val);12
Console.ReadLine(); // 让黑屏等待,不会直接关闭..13
}14
}15

16
public class AsyncTest17
{18
public string Hello(string name)19
{20
// 查看当前的线程ID, 是否线程池里面的线程21
Console.WriteLine("2,Thread ID:#{0},Is PoolThread?{1}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread);
22
return "Hello:" + name;23
}24
}

图1
我们可以从图1看出,我们平常写的Hello 程序是同一个线程的,而且不是线程池理的线程程序。按照上面的程序稍做改动, 那么开始我们第一个异步的Hello World 程序。
使用.Net 的委托机制来为我们的程序提供异步操作行为。
1步, 为我们的AsyncTest(Hello方法) 声明一个委托
public delegate string AsyncEventHandler(string name);
2步,使用委托提供的BeginInvoke, EndInvoke 方法(具体使用下一篇文章详细介绍)来提供异步的调用。
string val = test.Hello("Andy Huang");
修改为
AsyncEventHandler async = test.Hello;
IAsyncResult result = async.BeginInvoke("Andy Huang", null, null);
string val = async.EndInvoke(result);
下面是完整的代码.
Code 2:
class Program2
{3
static void Main(string[] args)4
{5
// 查看当前的线程ID, 是否线程池里面的线程6
Console.WriteLine("1,Thread ID:#{0},Is PoolThread?{1}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread);
7

8
AsyncTest test = new AsyncTest(); 9
//把Hello 方法分配给委托对象10
AsyncEventHandler async = test.Hello; //注释1,建议用=,不用+=11

12
//发起一个异步调用的方法,赋值"Andy Huang", 返回IAsyncResult 对象13
IAsyncResult result = async.BeginInvoke("Andy Huang", null, null);14

15
//这里会阻碍线程,直到方法执行完毕16
string val = async.EndInvoke(result);17

18
Console.WriteLine(val);19
Console.ReadLine(); // 让黑屏等待,不会直接关闭..20
}21
}22

23
//我们使用委托来提供.Net的异步机制24
public delegate string AsyncEventHandler(string name);25
public class AsyncTest26
{27
public string Hello(string name)28
{29
// 查看当前的线程ID, 是否线程池里面的线程30
Console.WriteLine("2,Thread ID:#{0},Is PoolThread?{1}", Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread);
31
return "Hello:" + name;32
}33
}
注释1: 用=操作符在于分配委托对象时候不需要初始化;并且异步调用时候只能有一个目标方法.
图2
对比图1 和图2, ,可以看出(2,Thread ID:#10,Is PoolThread?True),在使用异步机制的时候,其实就是开启一个新的线程来执行我们的方法,并且这个线程来自 线程堆. 到这里,我们就很好的解释了”什么是.Net异步机制?”
说到这里,结束了吗? 可能大家会想其实使用异步就是多了个委托( public delegate string AsyncEventHandler(string name);) 那么到底为我们做了些什么呢?
通过反编译(微软提供的IL Disassembler)我们看到

图3
编译器为我们生成了下面类定义(其实委托就是一个类)
Code 3
public sealed class AsyncEventHandler : MulticastDelegate
{
public AsyncEventHandler(object @object, IntPtr method)
{....}
public virtual IAsyncResult BeginInvoke(string name, AsyncCallback callback, object @object)
{...}
public virtual string EndInvoke(IAsyncResult result)
{...}
public virtual string Invoke(string name)
{...}
}
继承于MulticastDelegate, 提供了BeginInvoke / EndInvoke / Invoke.
Invoke 是我们Code 1 同步调用的方法,当然我们也可以直接使用Invoke来同步调用.
BeginInvoke / EndInvoke 是Code 2中使用的异步调用的方法,下篇文章我会详细介绍
