其中,某些調(diào)優(yōu)技術(shù)是在你的編程工作中實現(xiàn)的。而另一些技術(shù)是與應(yīng)用服務(wù)器的配置相關(guān)的。在本文中,我們將詳細(xì)地描述怎樣通過調(diào)整servlet和jsp頁面,來提高你的應(yīng)用程序的總體性能。在閱讀本文之前,假設(shè)你有基本的servlet和jsp的知識。
方法一:在servlet的init()方法中緩存數(shù)據(jù)
當(dāng)應(yīng)用服務(wù)器初始化servlet實例之后,為客戶端請求提供服務(wù)之前,它會調(diào)用這個servlet的init()方法。在一個servlet的生命周期中,init()方法只會被調(diào)用一次。通過在init()方法中緩存一些靜態(tài)的數(shù)據(jù)或完成一些只需要執(zhí)行一次的、耗時的操作,就可大大地提高系統(tǒng)性能。
例如,通過在init()方法中建立一個jdbc連接池是一個最佳例子,假設(shè)我們是用jdbc2.0的datasource接口來取得數(shù)據(jù)庫連接,在通常的情況下,我們需要通過jndi來取得具體的數(shù)據(jù)源。我們可以想象在一個具體的應(yīng)用中,如果每次sql請求都要執(zhí)行一次jndi查詢的話,那系統(tǒng)性能將會急劇下降。解決方法是如下代碼,它通過緩存datasource,使得下一次sql調(diào)用時仍然可以繼續(xù)利用它:
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提供了一個實用的技術(shù),即自動重載技術(shù),它為開發(fā)人員提供了一個好的開發(fā)環(huán)境,當(dāng)你改變servlet和jsp頁面后而不必重啟應(yīng)用服務(wù)器。然而,這種技術(shù)在產(chǎn)品運行階段對系統(tǒng)的資源是一個極大的損耗,因為它會給jsp引擎的類裝載器(classloader)帶來極大的負(fù)擔(dān)。因此關(guān)閉自動重載功能對系統(tǒng)性能的提升是一個極大的幫助。
方法 3: 不要濫用httpsession
在很多應(yīng)用中,我們的程序需要保持客戶端的狀態(tài),以便頁面之間可以相互聯(lián)系。但不幸的是由于http具有天生無狀態(tài)性,從而無法保存客戶端的狀態(tài)。因此一般的應(yīng)用服務(wù)器都提供了session來保存客戶的狀態(tài)。在jsp應(yīng)用服務(wù)器中,是通過httpsession對像來實現(xiàn)session的功能的,但在方便的同時,它也給系統(tǒng)帶來了不小的負(fù)擔(dān)。因為每當(dāng)你獲得或更新session時,系統(tǒng)者要對它進(jìn)行費時的序列化操作。你可以通過對httpsession的以下幾種處理方式來提升系統(tǒng)的性能。
如果沒有必要,就應(yīng)該關(guān)閉jsp頁面中對httpsession的缺省設(shè)置。 如果你沒有明確指定的話,每個jsp頁面都會缺省地創(chuàng)建一個httpsession。如果你的jsp中不需要使用session的話,那可以通過如下的jsp頁面指示符來禁止它:
<%@ page session="false"%>
不要在httpsession中存放大的數(shù)據(jù)對像:如果你在httpsession中存放大的數(shù)據(jù)對像的話,每當(dāng)對它進(jìn)行讀寫時,應(yīng)用服務(wù)器都將對其進(jìn)行序列化,從而增加了系統(tǒng)的額外負(fù)擔(dān)。你在httpsession中存放的數(shù)據(jù)對像越大,那系統(tǒng)的性能就下降得越快。
當(dāng)你不需要httpsession時,盡快地釋放它:當(dāng)你不再需要session時,你可以通過調(diào)用httpsession.invalidate()方法來釋放它。盡量將session的超時時間設(shè)得短一點:在jsp應(yīng)用服務(wù)器中,有一個缺省的session的超時時間。當(dāng)客戶在這個時間之后沒有進(jìn)行任何操作的話,系統(tǒng)會將相關(guān)的session自動從內(nèi)存中釋放。超時時間設(shè)得越大,系統(tǒng)的性能就會越低,因此最好的方法就是盡量使得它的值保持在一個較低的水平。
方法 4: 將頁面輸出進(jìn)行壓縮
壓縮是解決數(shù)據(jù)冗余的一個好的方法,特別是在網(wǎng)絡(luò)帶寬不夠發(fā)達(dá)的今天。有的瀏覽器支持gzip(gnu zip)進(jìn)行來對html文件進(jìn)行壓縮,這種方法可以戲劇性地減少html文件的下載時間。因此,如果你將servlet或jsp頁面生成的html頁面進(jìn)行壓縮的話,那用戶就會覺得頁面瀏覽速度會非常快。但不幸的是,不是所有的瀏覽器都支持gzip壓縮,但你可以通過在你的程序中檢查客戶的瀏覽器是否支持它。下面就是關(guān)于這種方法實現(xiàn)的一個代碼片段:
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: 使用線程池
應(yīng)用服務(wù)器缺省地為每個不同的客戶端請求創(chuàng)建一個線程進(jìn)行處理,并為它們分派service()方法,當(dāng)service()方法調(diào)用完成后,與之相應(yīng)的線程也隨之撤消。由于創(chuàng)建和撤消線程會耗費一定的系統(tǒng)資源,這種缺省模式降低了系統(tǒng)的性能。但所幸的是我們可以通過創(chuàng)建一個線程池來改變這種狀況。
另外,我們還要為這個線程池設(shè)置一個最小線程數(shù)和一個最大線程數(shù)。在應(yīng)用服務(wù)器啟動時,它會創(chuàng)建數(shù)量等于最小線程數(shù)的一個線程池,當(dāng)客戶有請求時,相應(yīng)地從池從取出一個線程來進(jìn)行處理,當(dāng)處理完成后,再將線程重新放入到池中。如果池中的線程不夠地話,系統(tǒng)會自動地增加池中線程的數(shù)量,但總量不能超過最大線程數(shù)。通過使用線程池,當(dāng)客戶端請求急劇增加時,系統(tǒng)的負(fù)載就會呈現(xiàn)的平滑的上升曲線,從而提高的系統(tǒng)的可伸縮性。
方法 6: 選擇正確的頁面包含機制
在jsp中有兩種方法可以用來包含另一個頁面:
1、使用include指示符
<%@ includee file=”test.jsp” %>
2、使用jsp指示符
<jsp:includee page=”test.jsp” flush=”true”/>
在實際中發(fā)現(xiàn),如果使用第一種方法的話,可以使得系統(tǒng)性能更高。
方法 7:正確地確定javabean的生命周期
jsp的一個強大的地方就是對javabean的支持。通過在jsp頁面中使用<jsp:usebean>標(biāo)簽,可以將javabean直接插入到一個jsp頁面中。它的使用方法如下:
<jsp:usebean id="name" scope="page|request|session|application" class=
"package.classname" type="typename">
</jsp:usebean>
其中scope屬性指出了這個bean的生命周期。缺省的生命周期為page。如果你沒有正確地選擇bean的生命周期的話,它將影響系統(tǒng)的性能。
舉例來說,如果你只想在一次請求中使用某個bean,但你卻將這個bean的生命周期設(shè)置成了session,那當(dāng)這次請求結(jié)束后,這個bean將仍然保留在內(nèi)存中,除非session超時或用戶關(guān)閉瀏覽器。這樣會耗費一定的內(nèi)存,并無謂的增加了jvm垃圾收集器的工作量。因此為bean設(shè)置正確的生命周期,并在bean的使命結(jié)束后盡快地清理它們,會使用系統(tǒng)性能有一個提高。
其它一些有用的方法
1、在字符串連接操作中盡量不使用“+”操作符:在java編程中,我們常常使用“+”操作符來將幾個字符串連接起來,但你或許從來沒有想到過它居然會對系統(tǒng)性能造成影響吧?由于字符串是常量,因此jvm會產(chǎn)生一些臨時的對像。你使用的“+”越多,生成的臨時對像就越多,這樣也會給系統(tǒng)性能帶來一些影響。解決的方法是用stringbuffer對像來代替“+”操作符。
2、避免使用system.out.println()方法:由于system.out.println()是一種同步調(diào)用,即在調(diào)用它時,磁盤i/o操作必須等待它的完成,因此我們要盡量避免對它的調(diào)用。但我們在調(diào)試程序時它又是一個必不可少的方便工具,為了解決這個矛盾,我建議你最好使用log4j工具(http://jakarta.apache.org ),它既可以方便調(diào)試,而不會產(chǎn)生system.out.println()這樣的方法。
3、servletoutputstream 與 printwriter的權(quán)衡:使用printwriter可能會帶來一些小的開銷,因為它將所有的原始輸出都轉(zhuǎn)換為字符流來輸出,因此如果使用它來作為頁面輸出的話,系統(tǒng)要負(fù)擔(dān)一個轉(zhuǎn)換過程。而使用servletoutputstream作為頁面輸出的話就不存在一個問題,但它是以二進(jìn)制進(jìn)行輸出的。因此在實際應(yīng)用中要權(quán)衡兩者的利弊。
總結(jié)
本文的目的是通過對servlet和jsp的一些調(diào)優(yōu)技術(shù)來極大地提高你的應(yīng)用程序的性能,并因此提升整個j2ee應(yīng)用的性能。通過這些調(diào)優(yōu)技術(shù),你可以發(fā)現(xiàn)其實并不是某種技術(shù)平臺(比如j2ee和.net之爭)決定了你的應(yīng)用程序的性能,重要是你要對這種平臺有一個較為深入的了解,這樣你才能從根本上對自己的應(yīng)用程序做一個優(yōu)化
新聞熱點
疑難解答