佳木斯湛栽影视文化发展公司

主頁 > 知識庫 > Tomcat中對靜態(tài)資源的處理教程

Tomcat中對靜態(tài)資源的處理教程

熱門標簽:網(wǎng)站建設 百度競價點擊價格的計算公式 阿里云 檢查注冊表項 使用U盤裝系統(tǒng) 美圖手機 智能手機 硅谷的囚徒呼叫中心

前言

Tomcat 中的請求都是由 Servlet 處理,靜態(tài)資源也不例外。在默認的 web.xml 中,配置了一個 DefaultServlet 用于處理靜態(tài)資源,它支持緩存和斷點續(xù)傳。

DefaultServlet 的基本處理過程如下:

  • 查找資源是否存在緩存
  • 檢查是否滿足可選 If 頭域指定的條件
  • 設置響應頭域,如 Content-Type、Content-Length、ETag、Last-Modified
  • 檢查是否滿足 Sendfile 的條件,否則將內(nèi)容拷貝到輸出流中

接下來主要分析資源緩存的設計和實現(xiàn),以及 If 頭域的處理。

1. 資源緩存的設計

訪問磁盤的速度遠遠低于訪問內(nèi)存的速度,所以適當?shù)木彺嬉徊糠朱o態(tài)資源能夠讓系統(tǒng)快速響應。

Tomcat 在 6.0.53 版本實現(xiàn)靜態(tài)資源的處理時,借助了 JNDI 的一些 API(但在使用時感覺與 JNDI 的關系不大),相關類圖及核心方法和屬性如下:

緩存相關的類:

  • ResourceCache: 緩存實現(xiàn),提供了資源查找、加載、銷毀的功能
  • CacheEntry: 一個緩存條目,包含緩存名稱,如 /tomcat.gif,資源和資源的屬性以及對應的目錄

資源目錄相關的類是:

  • EmptyDirContext: 主要用于嵌入式模式,行為就像沒有可用資源一樣
  • FileDirContext: 基于文件系統(tǒng)的資源目錄服務
  • WARDirContext: 基于 war 文件的目錄服務
  • Resource: 封裝了資源內(nèi)容,主要有字節(jié)數(shù)據(jù)和輸入流
  • ResourceAttributes: 資源屬性,主要有內(nèi)容長度和最后修改時間
  • ProxyDirContext: 資源緩存和目錄服務的代理,提供查找資源緩存、校驗緩存是否過期等功能

默認情況下,緩存最大為 10 MB,單個緩存資源最大為 512 KB,緩存的 TTL 為 5s。

一般的,在 Mapper 映射到處理靜態(tài)資源的 Wrapper 時,會引起資源的加載,基本的方法調(diào)用情況如下:

Mapper.map(MessageBytes, MessageBytes, MappingData)
└─Mapper.internalMap(CharChunk, CharChunk, MappingData)
 └─Mapper.internalMapWrapper(Mapper$Context, CharChunk, MappingData)
 └─ProxyDirContext.lookup(String)
 └─ProxyDirContext.cacheLookup(String)
 └─ResourceCache.lookup(String)
  └─ResourceCache.find(CacheEntry[], String)

緩存資源插入內(nèi)部數(shù)組時是有序的,find 方法就是通過資源名二分查找緩存,資源名就是請求路徑,此時有兩種情況,緩存命中和未命中。

緩存未命中,在 cacheLookup 方法中會新建一個 CacheEntry 對象,調(diào)用 cacheLoad 方法加入到 ResourceCache 的緩存數(shù)組中,加入前會對緩存條目進行以下操作:

  • 獲取并初始化緩存資源屬性,主要是文件的 contentLength 和 lastModified
  • 如果文件長度小于 512KB,那么將文件內(nèi)容加載到內(nèi)存中
  • 標記緩存存在,設置緩存時間戳

緩存命中,會對緩存條目進行校驗:

  • 檢查是否過期,當前時間大于緩存條目設置的時間戳
  • 如果過期,再檢查資源內(nèi)容是否修改
  • 如果修改,清除這個緩存,讀取最新內(nèi)容

以上就是資源緩存簡單的處理過程。

2. If 頭域的處理

客戶端接收并緩存請求的資源,,當再次請求此資源時,服務端根據(jù)特定的請求頭域來驗證資源是否修改,沒有變動,則只返回一個 304 Not Modified 響應,否則返回資源的內(nèi)容,從而節(jié)省帶寬。

用于資源驗證的頭域有兩種,分別是:Last-Modified+If-Modified-Since 和 ETag+If-None-Match。

Last-Modified+If-Modified-Since,單位是秒,這個容易理解,如果服務端資源的最后修改時間小于 If-Modified-Since 的值,表示資源無變動。與 If-Modified-Since 對應的有個 If-Unmodified-Since,它類似一個斷言,小于此時間戳的資源才返回,大于等于的話會返回 412 Precondition Failed 的錯誤。

使用時間戳校驗有幾個弊端:

  • 文件有可能只改變修改時間,內(nèi)容不變
  • 文件在秒以下的時間修改無法判斷
  • 服務器可能不能精確獲取文件的最后修改時間。

因此,HTTP 引入了 ETag。ETag(Entity Tags) 資源唯一標識,可看做服務端為資源生成的一個 Token,用于校驗資源是否修改。HTTP 只規(guī)定 ETag 要放在雙引號內(nèi),沒有規(guī)定內(nèi)容是什么或者要怎么實現(xiàn),Tomcat 生成 ETag 的邏輯是 "W/\"" + contentLength + "-" + lastModified + "\"" ,其中 'W/' 表示大小寫敏感。

ETag+If-None-Match,If-None-Match 的值由一個或多個 ETag 組成,多個以逗號分割,如果服務端資源的 ETag 與其中的任何一個都不匹配,表示請求的資源有修改;否則無變動。它還有一個特殊值-星號(*),只在資源上傳時使用,通常是 PUT 方法,檢查是否已經(jīng)上傳過。

此外 If-None-Match 的優(yōu)先級高于 If-Modified-Since,也就是說,存在 If-None-Match 就不對最后修改時間進行校驗。與 If-None-Match 相對的有個 If-Match,它也類似斷言,只有資源的 ETag 匹配時才認為沒有修改,通常用于斷點續(xù)傳。

Tomcat 實現(xiàn)此部分的核心代碼如下:

// 返回 true 是才認為資源有變動
protected boolean checkIfHeaders(HttpServletRequest request,
  HttpServletResponse response,ResourceAttributes resourceAttributes)
  throws IOException {
 return checkIfMatch(request, response, resourceAttributes)
  && checkIfModifiedSince(request, response, resourceAttributes)
  && checkIfNoneMatch(request, response, resourceAttributes)
  && checkIfUnmodifiedSince(request, response, resourceAttributes);
}

2.1 一次請求流程

以請求 /main.css 靜態(tài)資源為例,第一次請求響應頭信息如下:

HTTP/1.1 200 OK
Server: Apache-Coyote/1.1
Accept-Ranges: bytes
ETag: W/"72259-1557127244000"
Last-Modified: Mon, 06 May 2019 07:20:44 GMT
Content-Type: text/css
Content-Length: 72259
Date: Mon, 06 May 2019 07:20:57 GMT

第二次請求時,首先看一下請求頭域關鍵信息:

Cache-Control:max-age=0
Connection:keep-alive
Host:localhost:8080
If-Modified-Since:Mon, 06 May 2019 07:20:44 GMT
If-None-Match:W/"72259-1557127244000"

服務器收到請求后就會比對 ETag,這里匹配成功,表示資源沒有修改,響應如下:

HTTP/1.1 304 Not Modified
Server: Apache-Coyote/1.1
ETag: W/"72259-1557127244000"
Date: Mon, 06 May 2019 07:21:46 GMT

注意:在復現(xiàn)時,要使用文本類型,如果使用 Chrome 瀏覽器,記得開啟緩存。

2.2 Accept-Ranges

在上文的響應中,服務器設置了一個 Accept-Ranges: bytes 頭,字面理解就是可以請求資源的一部分字節(jié),客戶端發(fā)現(xiàn)有這個頭時,就可以嘗試斷點續(xù)傳。

解析過程就是對 HTTP 規(guī)范的實現(xiàn),這里不在具體分析了,規(guī)范詳細信息可查看 RFC7233#section-2.3.

3. SendFile 的處理

檢查是否支持 SendFile,NIO 模式下支持此操作,也就是零拷貝,此操作會減少一次到應用內(nèi)存的拷貝,直接從內(nèi)核將數(shù)據(jù)寫入通道。Tomcat 在文件大小大于 48KB 時會嘗試使用此方式發(fā)送。

4. 小結(jié)

Tomcat 對靜態(tài)資源處理的實現(xiàn)還是比較完善的,但還是略遜色于 Nginx 這類 Web 服務器,因為它們能直接處理靜態(tài)資源,而 Tomcat 還要多做一次映射。一般的都會進行動靜分離,讓 Tomcat 專注處理動態(tài)請求。

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。

標簽:通遼 煙臺 山南 賀州 湖北 黃山 湘潭 懷化

巨人網(wǎng)絡通訊聲明:本文標題《Tomcat中對靜態(tài)資源的處理教程》,本文關鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權問題,煩請?zhí)峁┫嚓P信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡,涉及言論、版權與本站無關。
  • 相關文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266
    台东县| 永胜县| 阿鲁科尔沁旗| 鄂托克前旗| 阜宁县| 隆化县| 论坛| 稻城县| 贵溪市| 泊头市| 南川市| 和顺县| 巴塘县| 彰武县| 丁青县| 定州市| 濮阳县| 麻江县| 通城县| 南丹县| 五寨县| 温州市| 南川市| 祁阳县| 康保县| 土默特右旗| 岳阳县| 眉山市| 株洲县| 明光市| 灯塔市| 涞水县| 民勤县| 潼南县| 尚志市| 全椒县| 兴隆县| 普格县| 若羌县| 如皋市| 新余市|