提升應用程序的整體性能七方法


在本文中,咱們將詳細地描述怎樣經過調整servlet和JSP頁面,來提升應用程序的整體性能。本文的目的是經過對servlet和JSP的一些調優技術來極大地提升你的應用程序的性能,並所以提高整個J2EE應用的性能。經過這些調優技術,你能夠發現其實並非某種技術平臺(好比J2EE和.NET之爭)決定了你的應用程序的性能,重要是你要對這種平臺有一個較爲深刻的瞭解,這樣你才能從根本上對本身的應用程序作一個優化。 
java

想學習更多關於java的知識,能夠點擊《Java EE軟件工程師》進行學習。
sql

方法一:在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;session

  public void init(ServletConfig config) throws ServletExceptionapp

{jsp

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的如下幾種處理方式來提高系統的性能。

若是沒有必要,就應該關閉JSP頁面中對HttpSession的缺省設置。 若是你沒有明確指定的話,每一個JSP頁面都會缺省地建立一個HttpSession。若是你的JSP中不須要使用session的話,那能夠經過以下的JSP頁面指示符來禁止它:

<%@ page session="false"%>


不要在HttpSession中存放大的數據對像:若是你在HttpSession中存放大的數據對像的話,每當對它進行讀寫時,應用服務器都將對其進行序列化,從而增長了系統的額外負擔。你在HttpSession中存放的數據對像越大,那系統的性能就降低得越快。

當你不須要HttpSession時,儘快地釋放它:當你再也不須要session時,你能夠經過調用HttpSession.invalidate()方法來釋放它。儘可能將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中有兩種方法能夠用來包含另外一個頁面:

一、使用include指示符

<%@ includee file=」test.jsp」 %>

二、使用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">

</jsp:useBean>

其中scope屬性指出了這個bean的生命週期。缺省的生命週期爲page。若是你沒有正確地選擇bean的生命週期的話,它將影響系統的性能。

舉例來講,若是你只想在一次請求中使用某個bean,但你卻將這個bean的生命週期設置成了session,那當此次請求結束後,這個bean將仍然保留在內存中,除非session超時或用戶關閉瀏覽器。這樣會耗費必定的內存,並沒有謂的增長了JVM垃圾收集器的工做量。所以爲bean設置正確的生命週期,並在bean的使命結束後儘快地清理它們,會使用系統性能有一個提升。

更多java學習課程,能夠關注e良師益友

相關文章
相關標籤/搜索