ASP.NET页面会被HTTP处理程序作为Page类的实例处理。每个请求会占用ASP.NET线程池中的一个线程,在请求完毕后该线程才会被释放。如果被请求的页面频繁地启动外部的、高耗时的任务时,经常会出现ASP.NET进程闲置,但池中没有空闲的线程来处理新入的其他页面的请求。这种情况下,创建异面页面可以减轻这个问题。
异步ASP.NET页面的构建涉及两个方面: @Page指令的一个新属性Async,以及注册若干异步执行的任务。异步任务可以通过两种途径注册:一种是为PreRenderComplete事件定义异步处理程序Begin/End对(AddOnPreRenderCompleteAsync方法用于将异步事件处理程序添加到页面的PreRenderComplete事件中);另一种是创建代表异步任务的PageAsyncTask对象,PageAsyncTask类代表要以异步方式执行任务(RegisterAsyncTask方法能够接受一个PageAsyncTask对象,返回void)。
AddOnPreRenderCompleteAsync和RegisterAsyncTask的区别:
从功能上讲,这两个方法几乎相同,二者都会将请求的执行分为两部分——分别在同步点前后。他们之间的区别在于:第一个区别在逻辑上,RegisterAsyncTask是一个用于在一个页面中运行多个异步任务的API(而不能跨越多个带有Async=true属性的异步页面),而AddOnPreRenderCompleteAsync是专门为多个异步页面设计的。RegisterAsyncTask执行End处理程序时,所处线程的上下文比AddOnPreRenderCompleteAsync所处线程的更丰富。该线程上下文包括模拟(impersonation)和HTTP上下文信息,该信息在一般异步页面的End处理程序所处线程中不存在。此外,RegisterAsyncTask还允许设置超时值,确保任何任务消耗的时间不会超过指定的秒数。另一个区别在:RegisterAsyncTask有利于实现对远程源的多个调用。我们只需设置一个布尔类型的标志,便会获得并行执行的能力,而不必要自行创建并管理自定义的IAsyncResult对象。
异步页面的加载分两个过程完成,第一部分是页面的PreInit 事件到PreRender 事件,第二部分开始执行异步任务,执行完异步任务后执行Page_PreRenderComplete事件,执行完后呈现html给浏览器。
前台代码示例:
View Code
<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="StreamDownload.aspx.cs" Inherits="WebTestDemo.AsyncPage.StreamDownload" Async="true" Trace="true"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
<title></title>
</head>
<body>
<form id="form1" runat="server">
<div>
<%=msdnData%>
</div>
</form>
</body>
</html>
注意:以上@Page指令的Async和Trace属性。Async属性使页面能够为PreRenderComplete事件注册异步处理程序,Trace属性使页面能够显示页面跟踪信息。
后台代码示例:
View Code
/// <summary>
/// 异步下载MSDN首页实例
/// </summary>
public partial class StreamDownload : System.Web.UI.Page
{
const string connectionToMSDN = "http://msdn.microsoft.com";
private WebRequest req;
public string msdnData;
protected override void OnInit(EventArgs e)
{
//将跟踪消息写入页面跟踪日志
Trace.Warn("当前线程:" + Thread.CurrentThread.ManagedThreadId.ToString());
}
protected void Page_Load(object sender, EventArgs e)
{
//AddOnPreRenderCompleteAsync方法
AddOnPreRenderCompleteAsync(
new BeginEventHandler(BeginTask),
new EndEventHandler(EndTask));
//RegisterAsyncTask方法
//PageAsyncTask task = new PageAsyncTask(
// new BeginEventHandler(BeginTask),
// new EndEventHandler(EndTask),
// null,
// null);
//RegisterAsyncTask(task);
}
IAsyncResult BeginTask(object sender, EventArgs e, AsyncCallback cb, object state)
{
Trace.Warn("开始异步 当前线程:" + Thread.CurrentThread.ManagedThreadId.ToString());
//准备一个Web请求
req = WebRequest.Create(connectionToMSDN);
//开始操作并返回IAsyncResult对象
return req.BeginGetResponse(cb, state);
}
void EndTask(IAsyncResult ar)
{
//这段代码将调用一个池里的线程
string text;
using (WebResponse response = req.EndGetResponse(ar))
{
StreamReader reader;
using (reader = new StreamReader(response.GetResponseStream()))
{
text = reader.ReadToEnd();
}
msdnData = ProcessFeed(text);
}
Trace.Warn("结束异步 当前线程:" + Thread.CurrentThread.ManagedThreadId.ToString());
}
protected override void OnSaveStateComplete(EventArgs e)
{
Trace.Warn("当前线程:" + Thread.CurrentThread.ManagedThreadId.ToString());
}
string ProcessFeed(string feed)
{
//来自XML输入,创建页面输出
return Server.HtmlDecode(feed);
}
}