高性能WEB開發 - JS、CSS的合並、壓縮、緩存管理


存在的問題:    

 

     合並、壓縮文件主要有2方面的問題:
       1. 每次發布的時候需要運行一下自己寫的bat文件或者其他程序把文件按照自己的配置合並和壓縮。
 
       2. 因生產環境和開發環境需要加載的文件不一樣,生產環境為了需要加載合並、壓縮后的文件,而開發環境為了修改、調試方便,需要加載非合並、壓縮的文件,所以我們常常需要在JSP中類似與下面的判斷代碼:
<c:if test="${env=='prod'}">
   <script type="text/javascript" src="/js/all.js"></script>
</c:if>
<c:if test="${env=='dev'}">
   <script type="text/javascript" src="/js/1.js"></script>
   <script type="text/javascript" src="/js/2.js"></script>
   <script type="text/javascript" src="/js/3.js"></script>
</c:if>

     

    緩存問題:在現在JS滿天飛的時代,大家都知道緩存能帶來的巨大好處,但緩存確實非常麻煩的一個問題,相信很多人曾經歷過下面的情況:為了讓程序更快,在服務器上為JS加上緩沖5天的代碼,但產品更新后第二天就接到電話說系統出錯,詳細了解后就發現是緩存引起的,讓用戶刪除緩存后就會OK。原因很簡單,就是你JS已經修改了,但用戶還在使用緩存中的老JS。在經歷幾次這種情況,被領導數落了幾次后。沒辦法只能把JS的緩沖去掉,或者改成8個小時。可這樣就完全失去了緩存的優勢了,哪我們到底需要解決哪些問題才能讓我們使用緩沖順心如意了?
    1. 如何在修改了某個JS后,自動把所有引用該JS頁面的代碼中加上1個版本號?

 

    2. 該如何生成版本號,根據什么來產生這個版本號。

 

    可能有人為了解決上面的緩存問題,寫了個JSP標簽,通過標簽讀取JS、css文件的修改時間來作為版本號,從而來解決上面2個問題。但這種方法有下面幾個缺點:
    1. 每次請求都要通過標簽讀取讀取文件的修改時間,速度慢。當然你可以把文件的修改時間放到緩存中,這樣也會加到了內存使用量。

 

    2. 在HTML靜態頁面中用不了

 

    3. 如果你們公司是如下的部署發布方式(我們公司就是這樣),則會失效。每次發布,不是直接覆蓋之前的WEB目錄,運維的為的發布方便,要求每次發布直接給他們1個war包,他們會把之前WEB目錄整個刪除,然后上傳現在的war包,這樣就導致程序運行后,所有文件的最后修改時間都是解壓war的時間。



分享自己項目中的處理方案:

      
    為了解決上面討論過的問題,在下寫了1個如下的組件,組件中根據我們自己的實際情況使用了文件大小來做為文件的版本號,雖然在文件修改很小(比如把字符a改成b),可能文件大小並沒有變,導致版本號也不會變。

 

但這種機率還是非常低的。當然如果你覺的使用文件修改時間作為版本號適合你,只需要修改一行代碼就行,下面看下這個組件的處理流程(本來想用流程圖表達,最后還是覺的文字來的直白寫):


    1. 程序啟動(contextInitialized)
 

    2. 搜索程序目錄下的所有merge.txt文件,根據merge.txt文件的配置合並文件, merge.txt文件實例如下:
# 文件合並配置文件,多個文件以|隔開,以/開頭的表示從根目錄開始, 
# 空格之后的文件名表示合並之后的文件名

# 把1,2,3合並到all文件中
1.js|2.js|3.js all.js

#合並CSS
/css/mian.css|/css/common.css all.css


    3. 搜索程序目錄下所有JS,CSS文件(包括合並后的),每個文件都壓縮后生成對應的1個新文件。


    4. 搜索程序目錄下所有JSP,html文件,把所有JS,css的引用代碼改成壓縮后並加了版本號的引用。

實例:

 

    實例的文件結構如下圖:
    

    

    看JSP原始代碼(程序運行前):
<%@page contentType="text/html" pageEncoding="UTF-8"%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"  "http://www.w3.org/TR/html4/loose.dtd">
<% boolean isDev = false;  // 是否開發環境%>
<html>
    <head>
        <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
        <title>JSP Page</title>
        <% if(isDev){ %>
        <script type="text/javascript" src="<%=request.getContextPath() %>/js/jquery-1.4.2.js"></script>
        <script type="text/javascript" src="<%=request.getContextPath() %>/js/1.js"></script>
        <script type="text/javascript" src="<%=request.getContextPath() %>/js/2.js"></script>
        <link type="text/css" rel="stylesheet" href="<%=request.getContextPath() %>/css/1.css" />
        <link type="text/css" rel="stylesheet" href="<%=request.getContextPath() %>/css/2.css" />
        <% }else{ %>
        <script type="text/javascript" src="<%=request.getContextPath() %>/js/jquery-1.4.2.js"></script>
        <script type="text/javascript" src="<%=request.getContextPath() %>/js/all.js"></script>
        <link type="text/css" rel="stylesheet"  href="<%=request.getContextPath() %>/css/all.css" />
        <% } %>
    </head>
    <body>
        <h1 class="c1">Hello World!</h1>
    </body>
</html>



    程序運行后JSP的代碼:
<%@page contentType="text/html" pageEncoding="UTF-8"%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<%
    boolean isDev = false;  // 是否開發環境
%>
<html>
    <head>
        <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
        <title>JSP Page</title>
        <% if(isDev){ %>
        <script type="text/javascript" src="<%=request.getContextPath() %>/js/jquery-1.4.2-3gmin.js?99375"></script>
        <script type="text/javascript" src="<%=request.getContextPath() %>/js/1-3gmin.js?90"></script>
        <script type="text/javascript" src="<%=request.getContextPath() %>/js/2-3gmin.js?91"></script>
        <link type="text/css" rel="stylesheet" href="<%=request.getContextPath() %>/css/1-3gmin.css?35" />
        <link type="text/css" rel="stylesheet" href="<%=request.getContextPath() %>/css/2-3gmin.css?18" />
        <% }else{ %>
        <script type="text/javascript" src="<%=request.getContextPath() %>/js/jquery-1.4.2-3gmin.js?99375"></script>
        <script type="text/javascript" src="<%=request.getContextPath() %>/js/all-3gmin.js?180"></script>
        <link type="text/css" rel="stylesheet"  href="<%=request.getContextPath() %>/css/all-3gmin.css?53" />
        <% } %>
    </head>
    <body>
        <h1 class="c1">Hello World!</h1>
    </body>
</html>

     加3gmin后綴的文件全部是程序啟動時自動生成的。

注意!

本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系我们删除。



 
粤ICP备14056181号  © 2014-2020 ITdaan.com