緩衝是Web應用中必須考慮的一個提高效能的重要手段。對於基於JSP/Servlet技術的網站,常用的緩衝有持久層的資料庫連接池緩衝,記憶體中的值對象緩衝,JSP頁面緩衝,以及各種各樣的緩衝架構等等,無不是為了提高系統的輸送量。
然而對於大型網站來說,將JSP頁面轉換為靜態Html也許是最高效的方法,特別適合於資料不經常變化但是頁面訪問量特別大的網站,如新聞等,通過把JSP動態網頁面預先轉換為靜態Html頁面,當使用者請求此頁面時,系統自動導向到對應的Html頁面,從而避免解析JSP請求,調用後台邏輯以及訪問資料庫等操作所帶來的巨大開銷。
如何將一個已有的JSP網站的動態JSP頁面轉化為靜態Html呢?我們希望在不用更改現有Servlet,JSP的前提下讓系統自動將這些JSP轉換為Html頁。幸運的是,Filter為我們提供了一種實現方案。
Filter是Servlet 2.2規範中最激動人心的特性。Filter能過濾特定URL如/admin/*並進行必要的預先處理,如修改Request和Response,從而實現定製的輸入輸出。更強大的是,Filter本身是一個責任鏈模式,它能一個接一個地傳遞下去,從而將實現不同功能的Filter串起來,並且可以動態組合。
要自動產生靜態頁面,用Filter截獲jsp請求並先進行預先處理,自動產生Html,是個不錯的主意。一個很容易想到的方法是在Filter截獲Request後,導向一個Servlet,在這個Servlet中向本機發送一個http請求,然後將響應寫入一個檔案:
URLConnection urlConn = URLConnection.open(http://localhost/req);
注意要避免遞迴。
另一個方法是不類比http,而是定製Response,把伺服器返回的JSP響應輸出到我們自己的Response中,就可以將響應快速寫入Html檔案,然後再發送給客戶。而且,由於沒有http類比請求,直接讀取伺服器響應速度非常快。
截獲Response的關鍵便是實現一個WrappedResponse,讓伺服器將響應寫入我們的WrappedResponse中。這類似於一個代理模式,Servlet 2.x已經提供了一個WrappedResponse類,我們只需要複寫其中的一些關鍵方法即可。
WrappedResponse實現了Response介面,它需要一個Response作為建構函式的參數,事實上這正是代理模式的應用:WrappedResponse充當了代理角色,它會將JSP/Servlet容器的某些方法調用進行預先處理,我們需要實現自己的方法。
綜上:用Filter實現HTML緩衝的步驟是:
1. 用Filter截獲請求,如/a.jsp?id=123,映射到對應的html檔案名稱為/html/a.jsp$id=123.htm。
2. 尋找是否有/html/a.jsp$id=123.htm,如果有,直接forward到此html,結束。
3. 如果沒有,實現一個WrappedResponse,然後調用filterChain(request, wrappedResponse)。
4. 將返回的WrappedResponse寫入檔案/html/a.jsp$id=123.htm,然後返迴響應給使用者。
5. 下一次使用者發送相同的請求時,到第2步就結束了。
使用這個方法的好處是不用更改現有的Servlet,JSP頁,限制是,JSP頁面結果不能與Session相關,需要登陸或使用者定製的頁面不能用這種方法緩衝。