程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> JSP編程 >> 關於JSP >> 為什麼JSP會比Beetl慢

為什麼JSP會比Beetl慢

編輯:關於JSP

為什麼JSP會比Beetl慢


許多人都不相信這個事實,作為前端常用渲染技術,JSP比Beetl慢。如果稍微了解這倆種技術的人,會分析:JSP是編譯成class的,而 Beetl總是解釋執行的。JSP肯定會比Beetl快。然而,事實並不是這樣,通過了許多性能測試,證明,Beetl還是要快的,如下是TEB模板引擎 性能基准測試結果:

可以看出,代表Beetl的綠色的,性能高於代表JSP的黃色大約2倍。

還有個帖子來自osc:http://my.oschina.net/tangcoffee/blog/303865,壓力測試jsp和beetl,證明beetl性能是JSP的2倍,如下是截取的部分數據

   采用jfinal+beetl模板,apache ab壓力測試結果

  • Time taken for tests:   0.656 seconds

  • Complete requests:      1000

  • Time per request:       32.813 [ms] (mean)

   未采用beetl,apache ab測試結果:

  • Time taken for tests:   1.297 seconds

  • Complete requests:      1000

  • Time per request:      64.844  [ms] (mean)

    究竟怎麼回事情,使得編譯執行的JSP執行比解釋執行的Beetl慢。基本上來說,Beetl並沒有做出超越常規的性能優化,而是JSP本身性能優化不夠導致的。

    第一: JSP對靜態文本處理的不夠好

如果你看看JSP編譯的後的java代碼(以Tocmat7為例),你會發現,JSP並沒有優化好靜態文本輸出。如下一個JSP代碼

  1. <%@ page language="java" contentType="text/html; charset=ISO-8859-1" 
  2.     pageEncoding="UTF-8"%> 
  3. <html> 
  4. <head> 
  5. <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1"> 
  6. <title>Test JSP</title> 
  7. </head> 
  8. <body> 
  9. <% 
  10. String a = "Test JSP"; 
  11. %> 
  12. <%=a %> 
  13. </body> 
  14. </html> 

Tomcat7 會編譯成為

  1. out.write("<html>\r\n"); 
  2.       out.write("<head>\r\n"); 
  3.       out.write("<meta http-equiv=\"Content-Type\" 
  4.             content=\"text/html; charset=ISO-8859-1\">\r\n"); 
  5.       out.write("<title>Test JSP</title>\r\n"); 
  6.       out.write("</head>\r\n"); 
  7.       out.write("<body>\r\n"); 
  8. String a = "Test JSP"; 
  9.       out.write('\r'); 
  10.       out.write('\n'); 
  11.       out.print(a ); 
  12.       out.write("\r\n"); 
  13.       out.write("</body>\r\n"); 
  14.       out.write("</html>"); 

    可以看出,對於靜態文本,JSP會多次調用out.write方法,而write方法內部,每次調用,都會做flush檢測等耗時機制。因此,更好的方式應該是將靜態文本合並一次性輸出,應該是下面這種會更好點

// 期望JSP的樣子

  1. out.write("<html>\r\n<head>\r\n ....<body>\r\n“); 
  2. String a = "Test JSP"; 
  3. out.write("\r\n“); 
  4. out.print(a ); 
  5. out.write("\r\n</body>\r\n</html>"); 

第二  就算JSP的實現做了如上更改,靜態文本處理還有優化空間。這是因為互聯網傳輸的二進制,因此會存在一個將靜態文本轉成 byte[] 輸出的過程,這是一個耗費CPU操作的過程,也就是JSP裡的write操作,內部還大量的編碼,而且,隨著JSP一次次渲染,編碼是一次一次重復,實驗 證明,這極大的降低了JSP性能。通過如下偽代碼可以驗證

  1. public static void main(String[] args)throws Exception { 
  2.   String text = "<html>你好中文!你好中文!你好中文!</html>"; 
  3.   { 
  4.    //模擬jsp 
  5.    long start = System.currentTimeMillis(); 
  6.    for(int i=0;i<1000000;i++){ 
  7.     byte[] bs = text.getBytes("UTF-8"); 
  8.     write(bs); 
  9.    } 
  10.    long end = System.currentTimeMillis(); 
  11.    System.out.println("jsp total="+(end-start)); 
  12.   } 
  13.  
  14.   { 
  15.    // 模擬beetl 
  16.    long start = System.currentTimeMillis(); 
  17.    byte[] bs = text.getBytes("UTF-8");   
  18.    for(int i=0;i<1000000;i++){   
  19.     write(bs); 
  20.    } 
  21.    long end = System.currentTimeMillis(); 
  22.    System.out.println("beetl total="+(end-start)); 
  23.   } 
  24.  
  25.  
  26. public static void write(byte[] bs){ 
  27.  

輸出是:

  • jsp total=228

  • beetl total=3

可見Beetl將靜態文本預先編碼成二進制,會提高性能很多。而通常JSP,總是靜態文本多過JSP Code的

第三,JSP在JSTL做的不夠完美,也導致性能很差

由於JSP是基於Java語言,語言本身是OO的,很多地方不適合模板場景使用,因此,自然而然采用JSTL來彌補JSP的不足,這也是後來很多項目都基本上采用了JSTL來寫模板。然而,JSTL的性能更加有問題。比如下一個簡單的JSTL判斷

  1. <c:choose> 
  2.    <c:when test="${param.newFlag== '1' || param.newFlag== '2'}">  
  3.     <th>1 or 2  <font color="Red">*</font> 
  4.    </c:when>  
  5. </c:choose> 

在我最初的想象裡,我認為jsp至少會編譯成如下代碼:

//期望JSP能編譯成如下代碼

  1. if(request.getParameter("newFlag").equals("1") 
  2.    ||request.getParameter("newFlag").equals("2")){ 
  3.  
  4.     out.print(...) 

但事實並不是這樣,對於如上JSTL,編譯成

  1. // 實際上JSP編譯的代碼 
  2. out.write((java.lang.String) 
  3.   org.apache.jasper.runtime.PageContextImpl.proprietaryEvaluate( 
  4. "${param.newFlag== '1' || param.newFlag== '2'}", 
  5. java.lang.String.class, 
  6. (javax.servlet.jsp.PageContext)_jspx_page_context, null, false)); 

也就是,JSP並沒有如預期的預編譯成java代碼,而是動態解釋執行了 test條件,這樣,性能不差才怪呢.

綜上所述,JSP之所以在基准測試還是實際的測試,都比Beetl慢不少,是因為他靜態文本輸出方面沒有去做積極的優化。像JSTL那樣的的解釋執行也極大的拖了JSP後退,而Beetl避免了這些問題。



  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved