posts - 36, comments - 419, trackbacks - 0, articles - 0
            BlogJava :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

          日歷

          <2025年6月>
          25262728293031
          1234567
          891011121314
          15161718192021
          22232425262728
          293012345

          搜索

          •  

          積分與排名

          • 積分 - 288795
          • 排名 - 200
               本篇文章主要討論下目前JS,CSS 合并、壓縮、緩存管理存在的一些問題,然后分享下自己項目中用到的1個處理方案,并提供1個實例下載。

          存在的問題:    
           
               合并、壓縮文件主要有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后綴的文件全部是程序啟動時自動生成的。

          實例下載:猛擊此處下載

          PS:自己的設計的處理方案并沒有解決"需要JSP中加判斷代碼的問題",主要是因為還沒有找到什么好的辦法去自動刪除1.js,2.js,3.js的3個引用,而插入1個新的all.js的引用,如果那位同學對解決這個問題有好的想法,請不吝賜教。
                如果有同學想使用這個組件,建議在測試環境下運行一次后,把修改后的程序直接上傳到正式服務器上,然后去掉這個功能,不然在服務器上每次啟動都調用這個功能還是需要花費一些時間和資源的 
                 其實一直想使用SVN中的版本號來控制緩存,這個是最嚴謹的一個方法,但也因為做起來太復雜,所以一直也沒做起來,以后以后有時間可以再研究。

              有需要請查看:高性能WEB開發系列


          [作者]:BearRui(AK-47)
          [博客]: http://www.aygfsteel.com/bearrui/
          [聲明]:本博所有文章版權歸作者所有(除特殊說明以外),轉載請注明出處.
          英雄,別走啊,幫哥評論下:  

          精彩推薦 好文要頂 水平一般 看不懂 還需努力

          評論

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理  回復  更多評論   

          2010-06-09 15:49 by panasia
          這個問題,我在csdn上問過。。到現在還是無人問津。。。當初我也是考慮用svn的版本號來解決這個問題。不過個人能力有限。。。。現在還未解決。

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理  回復  更多評論   

          2010-06-09 15:53 by panasia
          我們現在也碰到相同的問題。。。css,js經常性更新。。客戶端如果有緩存。。就讀不到最新的內容。。如果用js加當前時間來判斷。。每次客戶端都要下載。就不能緩存了。我想這個最好的辦法也只有用svn版本號來控制。。

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理  回復  更多評論   

          2010-06-09 15:56 by BearRui(AK-47)
          用SVN版本比較麻煩,目前我們用文件大小來做版本號幾乎不會再碰見緩存的問題了,感覺這種方法比較簡單易用。

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理[未登錄]  回復  更多評論   

          2010-06-17 00:05 by Charles
          這個問題確實很惡心啊,有一種做法就是用一個腳本去讀取所有文件的修改時間,你文中也提了,這種是自動化程度最高的一個做法,但是這種最麻煩的就是每次載入頁面,都要去讀取磁盤,這在訪問量大的時候,可能會帶來瓶頸,另一個問題就是文件修改時間確實也不算準。

          我現在想做的一個方案就是半自動化的,基本上是通過使用部署腳本,每次部署的時候,執行一次,也即沒部署一次,就執行一次文件歸并。但是沒有了自動化,也很無聊。

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理  回復  更多評論   

          2010-06-17 08:49 by BearRui(AK-47)
          不需要每次載入頁面的時候去讀取文件屬性,使用我文中的代碼,在程序啟動的時候執行一次就可以,算是自動化的。

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理  回復  更多評論   

          2010-06-24 11:56 by ed hardy
          軟件類的知識太深奧了,看懂它要有一段時間嘍!

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理[未登錄]  回復  更多評論   

          2010-08-10 11:02 by Allen
          我們的思路是:部署兩個war,一個war專門負責導航(其web-context不能修改,終端用戶使用此作為入口登錄),另一個war是真正的應用部署,每次從新部署,修改其web-context,這樣web資源自然會從新下載。

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理  回復  更多評論   

          2010-08-10 11:04 by BearRui(AK-47)
          @Allen
          哪你這樣是不是不每次重新部署,所有WEB資源都重新下載,而不是只下載修改了的文件?

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理  回復  更多評論   

          2010-11-17 15:52 by study
          版本號用MD5生成有啥害處么?

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理  回復  更多評論   

          2010-11-17 15:55 by BearRui(AK-47)
          @study
          MD5 好處在哪了?除了每次MD5消耗性能外

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理  回復  更多評論   

          2011-03-08 12:06 by 路過
          @panasia
          可以用文件的最后一次修改時間,來設置 Last-Modified

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理  回復  更多評論   

          2012-09-05 13:14 by dohkoos
          [quote]因生產環境和開發環境需要加載的文件不一樣,生產環境為了需要加載合并、壓縮后的文件,而開發環境為了修改、調試方便,需要加載非合并、壓縮的文件,所以我們常常需要在JSP中類似與下面的判斷代碼。[/quote]

          這個是開發流程的問題。設置兩個分支,prod和dev,分別使用不同的配置文件,在發布的時候打包就可以了。

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理  回復  更多評論   

          2014-06-26 11:27 by newobj
          第二個jar包的源碼能否分享下.
          還有替換jsp,html,js 中應用的腳本JsCssReplace中的replace方法還是不要替換的為好,因為如果引用其他網站上的腳本,也會自動加上-3gmin,就會找不到文件的.

          # re: 高性能WEB開發(7) - JS、CSS的合并、壓縮、緩存管理  回復  更多評論   

          2014-08-17 01:01 by andrei
          可以考慮SSI服務端包含,把JavaScript的引用(包含時間戳,版本)寫在SSI的文件中,到時候發布只發布SSI包含的文件就行了,然后環境分離的問題也迎刃而解。
          主站蜘蛛池模板: 长海县| 宽城| 阜新| 乌审旗| 乌拉特后旗| 泰顺县| 中阳县| 嘉峪关市| 界首市| 筠连县| 抚松县| 临海市| 庆云县| 桓仁| 宜川县| 岳普湖县| 成安县| 精河县| 山丹县| 遵义市| 桑植县| 东乡县| 杭锦后旗| 玉环县| 彩票| 南岸区| 永嘉县| 黔江区| 普陀区| 美姑县| 丰镇市| 衡阳市| 临漳县| 连平县| 桃园市| 山阴县| 砀山县| 绥宁县| 盐边县| 新竹市| 永济市|