ASP.NET 2.0的異步頁面刷新功能真得很酷,今天嘗試用它解決我的一個應用問題,真是幫了大忙了。
我的頁面要向後台的socket服務提交一個查詢,首先這是一個可能長時間不返回的操作;其次為了拿到socket服務返回的長度不定、有可能很大的Response數據,我不得不采用了dotNET的IAsyncResult BeginReceive(byte[] buffer, int offset, int size, SocketFlags socketFlags, AsyncCallback callback, object state);辦法來異步提交socket命令,然後在回調函數中迭代調用int EndReceive(IAsyncResult asyncResult);來獲取Response數據直到數據被取完,這一系列的數據片斷都緩存在一個MemoryBlock鏈表中。
這種情況下,ASP.NET頁面要想用表格展示Response數據,必須能夠被異步刷新。
ASP.NET 1.x 本質上不支持異步頁,但是通過堅韌的努力和不懈地創新可以生成異步頁。有關更多概述信息,請參閱 MSDN®Magazine 2003 年 6 月刊的文章“Use Threads and Build Asynchronous Handlers in Your Server-Side Web Code”,該文章的作者是 Fritz Onion。
但ASP.NET 2.0 極大地簡化了生成異步頁的方式。首先使用該頁的 @ Page 指令引入 Async=“true” 屬性。
其次,在 Page_Load 時,注冊事件Page_PreRender:
private void Page_Load(object sender, System.EventArgs e)
{
if (!IsPostBack)
{
// 注冊事件Page_PreRender執行完成時執行方法
// Hook PreRenderComplete event for data binding
this.PreRenderComplete += new EventHandler(Page_PreRenderComplete);
接下來,我在點擊按鈕的事件函數中,開始我的異步操作之旅:
/// PageAsyncTask 的方式則是增強版本,除了異步頁面處理開始和結束方法自身外,
/// 還可以提供在超時情況下的處理方法,以及處理時的狀態對象。
///
PageAsyncTask task = new PageAsyncTask(
new BeginEventHandler(BeginSubmitKeywordsAsyncOperation),
new EndEventHandler(EndSubmitKeywordsAsyncOperation),
new EndEventHandler(TimeoutSubmitKeywordsAsyncOperation),
null
);
RegisterAsyncTask(task);
/// 對大多數簡單的異步處理情況,可以直接調用AddOnPreRenderCompleteAsync 方法,
/// 提交頁面請求開始和結束時的處理代碼
//AddOnPreRenderCompleteAsync(
// new BeginEventHandler(BeginSubmitKeywordsAsyncOperation),
// new EndEventHandler(EndSubmitKeywordsAsyncOperation)
//);
該頁經歷其常規處理生命周期,直到 PreRender 事件剛剛引發之後。然後,ASP.NET 調用使用 RegisterAsyncTask注冊的 Begin 方法BeginSubmitKeywordsAsyncOperation。這個方法中我將向sokcet服務提交我的命令,然後將socket.BeginReceive方法得到的IasyncResult對象返回給ASP.NET。
然後,我的socket異步回調函數在最終全部獲得數據流後,通過調用
callback.Invoke(result);
來通知頁面socket這邊已經完成工作了。
於是ASP.NET喚起EndSubmitKeywordsAsyncOperation方法:
/// <summary>
/// 異步調用結束後的接收方法(異步操作執行完成後,會重新從線程池中取個線程為本頁面請求服務).
/// </summary>
/// <param name="ar"></param>
void EndSubmitKeywordsAsyncOperation(IAsyncResult ar)
{
_resultDataset = _submit.getResultDataset(ar);
在這個函數中,我就可以拿到數據流了。
之後,ASP.NET又喚起Page_PreRender事件,從而可以在這個事件中,把頁面上的表格正確填充了。
這樣,就達到了我的目的。而且向後台提交後等待被再度激活超時了,也可以由
void TimeoutSubmitKeywordsAsyncOperation(IAsyncResult ar)
來做合適的處理。