首先,實現IHttpHandler時要實現一個IsReusable的屬性,這個屬性告訴ASP.NET此HttpHandler是否可重用。如果一個HttpHandler是可重用的,那麼多次請求都有可能用同一個HttpHandler實例;而如果一個HttpHandler是不可重用的,那麼ASP.Net應該確保每次請求使用的都是一個新構造的HttpHandler實例。
Page是設計為不可重用,所以每次請求都會導致構造一個新的Page實例,這是因為Page的生命周期不能恢復到初始狀態,一個Page經歷完生命周期後就不能用於處理下一次的請求。類似的,如果我們有一個HttpHandler有類似的性質,處理一次請求後其狀態就難以恢復到適合於處理下一次請求,或者說恢復還不如構造一個新的,那麼我們就應該設計為不可重用。
我在做一個通過IFrame提交的無刷新上傳控件,這東西包括一些HttpHandler,為的是能夠直接關聯到axd後綴而用於路徑無關的場合。其中有一個HttpHandler我直接繼承自Page,並且寫得好像ASPx+cs編譯出來的代碼那樣,在OnInit階段構建完整的控件樹。這個HttpHandler以前在XP的IIS5上一直沒問題的,但到了Vista的IIS7就出問題了。
先說明,在IIS7我采用其新的配置模式,將<httpHandlers />配置在<system.webServer />節,而不是<system.web />節,這是模仿著ASP.Net AJax的web.config做的。做好之後就發現問題了,這個用作HttpHandler的Page竟然時候拋出Exception,說Page上面有多於一個的HtmlForm。仔細檢查後確認,我的Page代碼確實僅僅添加了一個HtmlForm,並且這個Exception不會出現在編譯後的第一次請求,於是我就懷疑Page被重用了,所以OnInit被多次執行,這才可能導致它有多於一個HtmlForm。我啟用了Page的Trace,在Render中注釋掉base.Render,並且用Trace輸出Page上的HtmlForm數量,發現真的是每次請求都會導致多一個HtmlForm,這基本上可以肯定是因為Page被重用了。
然後我就用Google搜索,結果發現forums.ASP.net上有人提出了完全一樣的問題,他也是用Page來做HttpHandler。我想只有拿Page來做HttpHandler的人才會遇到這樣的問題,因為一般自己寫的HttpHandler都是無狀態的,所以都是可以重用的。而那張帖子只有管理員回復了一句,“你最好去forums.iis.net問吧”。於是我就去iis.net搜索,結果發現沒有人提到過這個問題,於是只好自己去論壇提問,可惜等了一天都沒有人回答,看來IIS7普及之前iis.net的人氣都不會上升(IIS7的默認歡迎頁面鏈接到IIS.Net)。
最後,我選擇了先采用Jeffrey Zhao建議的work around,就是再制作一個HttpHandlerFactory,它負責每次返回Page的新實例,然後在配置中改用該HttpHandlerFactory。這個解決方案實驗證明是可行的,就是多一個類而已,不知道性能損失有多少。如果有人知道這個問題的官方解決方案的話,或者有一個更好的work around,請告訴我,謝謝。