溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

JSP和Servlet性能優(yōu)化的方法有哪些

發(fā)布時間:2021-11-19 10:54:02 來源:億速云 閱讀:157 作者:小新 欄目:編程語言

小編給大家分享一下JSP和Servlet性能優(yōu)化的方法有哪些,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

方法一:在HttpServlet init()方法中緩存數(shù)據(jù)

服務(wù)器會在創(chuàng)建servlet實例之后和servlet處理任何請求之前調(diào)用servlet的init()方法。該方法在servlet的生命周期中僅調(diào)用一次。為了提高性能,在init()中緩存靜態(tài)數(shù)據(jù)或完成要在初始化期間完成的代價昂貴的操作。例如,一個***實踐是使用實現(xiàn)了javax.sql.DataSource接口的JDBC連接池。

DataSource從JNDI樹中獲得。每調(diào)用一次SQL就要使用JNDI查找DataSource是非常昂貴的工作,而且嚴(yán)重影響了應(yīng)用的性能。Servlet的init()方法可以用于獲取DataSource并緩存它以便之后的重用:

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; 
} 
... 
... 
}

方法二:禁用servlet和Jsp的自動裝載功能

當(dāng)每次修改了Servlet/JSP之后,你將不得不重新啟動服務(wù)器。由于自動裝載功能減少開發(fā)時間,該功能被認(rèn)為在開發(fā)階段是非常有用的。但是,它在運(yùn)行階段是非常昂貴的;servlet/JSP由于不必要的裝載,增加類裝載器的負(fù)擔(dān)而造成很差的性能。同樣,這會使你的應(yīng)用由于已被某種類裝載器裝載的類不能和當(dāng)前類裝載器裝載的類不能相互協(xié)作而出現(xiàn)奇怪的沖突現(xiàn)象。因此,在運(yùn)行環(huán)境中為了得到更好的性能,關(guān)閉servlet/JSP的自動裝載功能。

方法三:控制HttpSession

許多應(yīng)用需要一系列客戶端的請求,因此他們能互相相關(guān)聯(lián)。由于HTTP協(xié)議是無狀態(tài)的,所以基于Web的應(yīng)用需要負(fù)責(zé)維護(hù)這樣一個叫做session的狀態(tài)。為了支持必須維護(hù)狀態(tài)的應(yīng)用,Java servlet技術(shù)提供了管理session和允許多種機(jī)制實現(xiàn)session的API。HttpSession對象扮演了session,但是使用它需要成本。無論何時HttpSession被使用和重寫,它都由servlet讀取。你可以通過使用下面的技術(shù)來提高性能:

在JSP頁面中不要創(chuàng)建默認(rèn)的HttpSession:默認(rèn)情況下,JSP頁面創(chuàng)建HttpSession。如果你在JSP頁面中不用HttpSession,為了節(jié)省性能開銷,使用下邊的頁面指令可以避免自動創(chuàng)建HttpSession對象:

< %@ page session="false"%>

不要將大的對象圖存儲在HttpSession中:如果你將數(shù)據(jù)當(dāng)作一個大的對象圖存儲在HttpSession中,應(yīng)用服務(wù)器每次將不得不處理整個HttpSession對象。這將迫使Java序列化和增加計算開銷。由于序列化的開銷,隨著存儲在HttpSession對象中數(shù)據(jù)對象的增大,系統(tǒng)的吞吐量將會下降。

用完后釋放HttpSession:當(dāng)不在使用HttpSession時,使用HttpSession.invalidate()方法使sesion失效。

設(shè)置超時值:一個servlet引擎有一個默認(rèn)的超時值。如果你不刪除session或者一直把session用到它超時的時候,servlet引擎將把session從內(nèi)存中刪除。由于在內(nèi)存和垃圾收集上的開銷,session的超時值越大,它對系統(tǒng)彈性和性能的影響也越大。試著將session的超時值設(shè)置的盡可能低。

方法四:使用gzip壓縮

壓縮是刪除冗余信息的作法,用盡可能小的空間描述你的信息。使用gzip(GNU zip)壓縮文檔能有效地減少下載HTML文件的時間。你的信息量越小,它們被送出的速度越快。因此,如果你壓縮了由你web應(yīng)用產(chǎn)生的內(nèi)容,它到達(dá)用戶并顯示在用戶屏幕上的速度就越快。不是任何瀏覽器都支持gzip壓縮的,但檢查一個瀏覽器是否支持它并發(fā)送gzip壓縮內(nèi)容到瀏覽器是很容易的事情。下邊的代碼段說明了如何發(fā)送壓縮的內(nèi)容。

public void doGet(HttpServletRequest request, HttpServletResponse response) 
throws IOException, ServletException 
{ 
OutputStream out = null 
// Check the Accepting-Encoding header from the HTTP request. 
// If the header includes gzip, choose GZIP. 
// If the header includes compress, choose ZIP. 
// Otherwise choose no compression. 
String encoding = request.getHeader("Accept-Encoding"); 
if (encoding != null && encoding.indexOf("gzip") != -1) 
{ 
response.setHeader("Content-Encoding" , "gzip"); 
out = new GZIPOutputStream(response.getOutputStream()); 
} 
else if (encoding != null && encoding.indexOf("compress") != -1) 
{ 
response.setHeader("Content-Encoding" , "compress"); 
out = new ZIPOutputStream(response.getOutputStream()); 
} 
else 
{ 
out = response.getOutputStream(); 
} 
... 
... 
}

方法五:不要使用SingleThreadModel

SingleThreadModel保證servlet一次僅處理一個請求。如果一個servlet實現(xiàn)了這個接口,servlet引擎將為每個新的請求創(chuàng)建一個單獨(dú)的servlet實例,這將引起大量的系統(tǒng)開銷。如果你需要解決線程安全問題,請使用其他的辦法替代這個接口。SingleThreadModel在Servlet 2.4中是不再提倡使用。

方法六:使用線程池

servlet引擎為每個請求創(chuàng)建一個單獨(dú)的線程,將該線程指派給service()方法,然后在service()方法執(zhí)行完后刪除該線程。默認(rèn)情況下,servlet引擎可能為每個請求創(chuàng)建一個新的線程。由于創(chuàng)建和刪除線程的開銷是很昂貴的,于是這種默認(rèn)行為降低了系統(tǒng)的性能。我們可以使用線程池來提高性能。根據(jù)預(yù)期的并發(fā)用戶數(shù)量,配置一個線程池,設(shè)置好線程池里的線程數(shù)量的最小和***值以及增長的最小和***值。起初,servlet引擎創(chuàng)建一個線程數(shù)與配置中的最小線程數(shù)量相等的線程池。然后servlet引擎把池中的一個線程指派給一個請求而不是每次都創(chuàng)建新的線程,完成操作之后,servlet引擎把線程放回到線程池中。使用線程池,性能可以顯著地提高。如果需要,根據(jù)線程的***數(shù)和增長數(shù),可以創(chuàng)建更多的線程。

方法七:選擇正確的包括機(jī)制

在JSP頁面中,有兩中方式可以包括文件:包括指令(< %@ include file="test.jsp" %>)和包括動作(< jsp:include page="test.jsp" flush="true" />)。包括指令在編譯階段包括一個指定文件的內(nèi)容;例如,當(dāng)一個頁面編譯成一個servlet時。包括動作是指在請求階段包括文件內(nèi)容;例如,當(dāng)一個用戶請求一個頁面時。包括指令要比包括動作快些。因此除非被包括的文件經(jīng)常變動,否則使用包括指令將會獲得更好的性能。

方法八:在useBean動作中使用合適的范圍

使用JSP頁面***大方式之一是和JavaBean組件協(xié)同工作。JavaBean使用<jsp:useBean>標(biāo)簽可以嵌入到JSP頁面中。語法如下:

<jsp:useBean id="name" scope="page|request|session|application" class= 
"package.className" type="typeName">
</jsp:useBean>

scope屬性說明了bean的可見范圍。scope屬性的默認(rèn)值是page。你應(yīng)該根據(jù)你應(yīng)用的需求選擇正確的范圍,否則它將影響應(yīng)用的性能。

例如,如果你需要一個專用于某些請求的對象,但是你把范圍設(shè)置成了session,那么那個對象將在請求結(jié)束之后還保留在內(nèi)存中。它將一直保留在內(nèi)存中除非你明確地把它從內(nèi)存中刪除、使session無效或session超時。如果你沒有選擇正確的范圍屬性,由于內(nèi)存和垃圾收集的開銷將會影響性能。因此為對象設(shè)置合適的范圍并在用完它們之后立即刪除。

以上是“JSP和Servlet性能優(yōu)化的方法有哪些”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI