本文介紹了幾個調整JSP和servlet的一些非常實用的方法,它可使你的servlet和JSP頁面響應更快,擴展性更強。而且在用戶數增加的情況下,系統負載會呈現出平滑上長的趨勢。我將通過一些實際例子和配置方法使得你的應用程序的性能有出人意料的提升。其中,某些調優技術是在你的編程工作中實現的,而另一些技術是與應用服務器的配置相關的。我們將詳細地描述怎樣通過調整servlet和JSP頁面,來提高你的應用程序的總體性能。在閱讀本文之前,你還需要有基本的servlet和JSP的知識。
方法一:在servlet的init()方法中緩存數據
當應用服務器初始化servlet實例之後,為客戶端請求提供服務之前,它會調用這個servlet的init()方法。在一個servlet的生命周期中,init()方法只會被調用一次。通過在init()方法中緩存一些靜態的數據或完成一些只需要執行一次的、耗時的操作,就可大大地提高系統性能。
例如,通過在init()方法中建立一個JDBC連接池是一個最佳例子,假設我們是用jdbc2.0的DataSource接口來取得數據庫連接,在通常的情況下,我們需要通過JNDI來取得具體的數據源。我們可以想象在一個具體的應用中,如果每次SQL請求都要執行一次JNDI查詢的話,那系統性能將會急劇下降。解決方法是如下代碼,它通過緩存DataSource,使得下一次SQL調用時仍然可以繼續利用它:
public class ControllerServlet
extends HttpServlet
{
private javax.sql.DataSource
testDS = null;
public void init(ServletConfig config)
throws ServletException
{
super.init(config);
Context ctx = null;
try
{
ctx = new InitialContext();
testDS = (javax.sql.DataSource)
ctx.lookup("jdbc/testDS");
}
catch(NamingException ne)
{
ne.printStackTrace();
}
catch(Exception e)
{
e.printStackTrace();
}
}
public javax.sql.DataSource getTestDS()
{
return testDS;
}
...
...
}
方法 2:禁止servlet和JSP 自動重載(auto-reloading)
Servlet/JSP提供了一個實用的技術,即自動重載技術,它為開發人員提供了一個好的開發環境,當你改變servlet和JSP頁面後而不必重啟應用服務器。然而,這種技術在產品運行階段對系統的資源是一個極大的損耗,因為它會給JSP引擎的類裝載器(classloader)帶來極大的負擔。因此關閉自動重載功能對系統性能的提升是一個極大的幫助。
方法 3: 不要濫用HttpSession
在很多應用中,我們的程序需要保持客戶端的狀態,以便頁面之間可以相互聯系。但不幸的是由於HTTP具有天生無狀態性,從而無法保存客戶端的狀態。因此一般的應用服務器都提供了session來保存客戶的狀態。
在JSP應用服務器中,是通過HttpSession對像來實現session的功能的,但在方便的同時,它也給系統帶來了不小的負擔。因為每當你獲得或更新session時,系統者要對它進行費時的序列化操作。你可以通過對HttpSession的以下幾種處理方式來提升系統的性能:
1、如果沒有必要,就應該關閉JSP頁面中對HttpSession的缺省設置。如果你沒有明確指定的話,每個JSP頁面都會缺省地創建一個HttpSession。如果你的JSP中不需要使用session的話,那可以通過如下的JSP頁面指示符來禁止它:
<%@ page session="false"%>
2、不要在HttpSession中存放大的數據對像:如果你在HttpSession中存放大的數據對像的話,每當對它進行讀寫時,應用服務器都將對其進行序列化,從而增加了系統的額外負擔。你在HttpSession中存放的數據對像越大,那系統的性能就下降得越快。
3、當你不需要HttpSession時,盡快地釋放它:當你不再需要session時,你可以通過調用HttpSession.invalidate()方法來釋放它。
4、盡量將session的超時時間設得短一點:在JSP應用服務器中,有一個缺省的session的超時時間。當客戶在這個時間之後沒有進行任何操作的話,系統會將相關的session自動從內存中釋放。超時時間設得越大,系統的性能就會越低,因此最好的方法就是盡量使得它的值保持在一個較低的水平。
方法 4: 將頁面輸出進行壓縮
壓縮是解決數據冗余的一個好的方法,特別是在網絡帶寬不夠發達的今天。有的浏覽器支持gzip(GNU zip)進行來對HTML文件進行壓縮,這種方法可以戲劇性地減少HTML文件的下載時間。因此,如果你將servlet或JSP頁面生成的HTML頁面進行壓縮的話,那用戶就會覺得頁面浏覽速度會非常快。但不幸的是,不是所有的浏覽器都支持gzip壓縮,但你可以通過在你的程序中檢查客戶的浏覽器是否支持它。下面就是關於這種方法實現的一個代碼片段:
public void doGet(HttpServletRequest request,
HttpServletResponse response)
throws IOException,
ServletException
{
OutputStream out = null
String encoding =
request.getHeader("Accept-Encoding");
if (encoding !=
null && encoding.indexOf("gzip") != -1)
{
request.setHeader
("Content-Encoding" , "gzip");
out = new GZIPOutputStream
(request.getOutputStream());
}
else if (encoding !=
null && encoding.indexOf("compress") != -1)
{
request.setHeader
("Content-Encoding" , "compress");
out = new ZIPOutputStream
(request.getOutputStream());
}
else
{
out = request.getOutputStream();
}
...
...
}
方法 5: 使用線程池
應用服務器缺省地為每個不同的客戶端請求創建一個線程進行處理,並為它們分派service()方法,當service()方法調用完成後,與之相應的線程也隨之撤消。由於創建和撤消線程會耗費一定的系統資源,這種缺省模式降低了系統的性能。但所幸的是我們可以通過創建一個線程池來改變這種狀況。
另外,我們還要為這個線程池設置一個最小線程數和一個最大線程數。在應用服務器啟動時,它會創建數量等於最小線程數的一個線程池,當客戶有請求時,相應地從池從取出一個線程來進行處理,當處理完成後,再將線程重新放入到池中。
如果池中的線程不夠地話,系統會自動地增加池中線程的數量,但總量不能超過最大線程數。通過使用線程池,當客戶端請求急劇增加時,系統的負載就會呈現的平滑的上升曲線,從而提高的系統的可伸縮性。
方法 6: 選擇正確的頁面包含機制
在JSP中有兩種方法可以用來包含另一個頁面:
1、使用include指示符(<%@ includee file=”test.jsp” %>)。
2、使用jsp指示符(<jsp:includee page=”test.jsp” flush=”true”/>)。
在實際中我發現,如果使用第一種方法的話,可以使得系統性能更高。
方法 7:正確地確定javabean的生命周期
JSP的一個強大的地方就是對javabean的支持。通過在JSP頁面中使用<jsp:useBean>標簽,可以將javabean直接插入到一個JSP頁面中。它的使用方法如下:
<jsp:useBean id="name"
scope="page|request|session|application"
class=
"package.className" type="typeName&quo