首先,實現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,請告訴我,謝謝。
|
新聞熱點
疑難解答
圖片精選