代码改变世界

Make Asynchronous Calls from Page

2009-04-06 14:22  Jun1st  阅读(1495)  评论(0编辑  收藏  举报

 

对于ASP.NET pages的请求需要有HTTP handler来处理,比如Page的instance。当有一个request来请求某个.aspx页面时,ASP.NET runtime会从ASP.NET thread pool中选一个thread来处理这个request, 并且在这个请求处理完成时释放这个thread。因此,如果这个  request请求的页面中的某一个Step花费了相对较长的时间,比如调用某个web service来获取数据,那么结果就是处理这个请求的thread会一直处于idle的状态,而不能处理新的请求。而之所以这个thread会处于idle的状态,是由于ASP.NET handler处理请求时采用的是同步的方式。

为了解决这个问题,在ASP.NET 1.0中可以通过实现IHTTPAsyncHandler来实现异步的方式。而在ASP.NET 2.0之后,实现异步方式变得了更加的简单了。本文就介绍两种实现异步的方法。

 

1. Asynchronous Pages

要将一个普通的ASP.NET页面变成一个asynchronous page,只需要在 @Page directive 设置 Async=”true 就可以了。

<%@ Page Async="true" %>

 

除此之外,还需要在Page_Load事件中来注册需要完成的异步事件。在Asynchronous page中,可以通过调AddOnPreRenderCompleteAsync这个方法来注册,注册一对Begin/End的event handler。

1:  AddOnPreRenderCompleteAsync(
2:                  new BeginEventHandler(BeginTask),
3:                  new EndEventHandler(EndTask));

 

BeginEventHandler的方法签名

IAsyncResult BeginTask(object sender, EventArgs e, AsyncCallback cb, object state)

EndEventHandler的方法签名就相对的简单了

void EndTask(IAsyncResult ar)

那么注册的异步事件是在什么时候被执行的呢?在PreRenderComplete之前你注册了异步事件,注册的BeginEventHandler事件就会开始执行。当异步事件执行完成后,Page就会从PreRenderComplete开始,继续完成生命周期中剩下的各个事件。为了验证异步事件是否确实是在这个PreRenderComplete事件中被执行的,我们可以再 Begin/End 事件中往页面的Trace中写入信息,并打开页面的Trace,然后看输出的Track信息

如下:

aspx.page

Begin PreRender

0.000143719091709968

0.000011

aspx.page

End PreRender

0.000170049306985077

0.000026

 

Begin asyc: Thread=6

0.00101590747269794

0.000846

 

End async: Thread=7

2.20676898637082

2.205753

aspx.page

Begin PreRenderComplete

2.20682859505263

0.000060

aspx.page

End PreRenderComplete

2.20684834271408

0.000020

从Trace信息中可以看出,注册的异步事件就是在PreRenderPreRenderComplete之间执行的。

 

2. RegisterAsyncTask 方法

除了AddOnPreRenderCompleteAsync之外,还可以用 RegisterAsyncTask 来注册异步事件。甚至,RegisterAsyncTask 也许是一个更好的方法。

PageAsyncTask task = new PageAsyncTask(
    new BeginEventHandler(BeginTask),
    new EndEventHandler(EndTask),
    null,
    null);
 
RegisterAsyncTask(task);

RegisterAsyncTask 接受一个PageAsyncTask参数,这个task参数中可以设置的除了 Begin/End 事件外还可以做一些别的设置,如timeout时的handler等。timeout的默认时间是45秒,这个时间可以在 @Page directive中和web.config文件中进行设置

 

3. 两者比较
  1. RegisterAsyncTask 只是一个API方法,在异步和同步的页面中都可以使用。而AddOnPreRenderCompleteAsync只能在异步页面中使用
  2. RegisterAsyncTask end handler执行时,能够获得更多的信息,如HTTP context 信息。
  3. 当执行单个任务时,两者几乎一样,而当需要执行多个task时,则应该选用RegisterAsyncTask ,因为它允许多个任务同时执行
4. Aysnc-Compliant Action

什么样的操作应该或者建议使用async的操作呢? 大致上, CPU bound 或者 I/O bound的操作可以考虑采用异步的方式。尤其是在有大量IO操作的方法。这时,大量的时间都花费在了读取/写入或者操作数据上,这时,cpu就处于idle的状态,等待IO操作的完成

 

Sample

个人浅见,如果不足之处请指正,谢谢