細說ASP.NET Core靜態文件的緩存方式

来源:http://www.cnblogs.com/wufei999/archive/2017/04/28/6782188.html
-Advertisement-
Play Games

一、前言 我們在優化Web服務的時候,對於靜態的資源文件,通常都是通過客戶端緩存、伺服器緩存、CDN緩存,這三種方式來緩解客戶端對於Web伺服器的連接請求壓力的。 本文指在這三個方面,在ASP.NET Core中靜態文件的實現過程和使用方法進行闡述。當然也可以考慮使用反向代理的方式(例如IIS或Ng ...


一、前言

  我們在優化Web服務的時候,對於靜態的資源文件,通常都是通過客戶端緩存伺服器緩存CDN緩存,這三種方式來緩解客戶端對於Web伺服器的連接請求壓力的。

  本文指在這三個方面,在ASP.NET Core中靜態文件的實現過程和使用方法進行闡述。當然也可以考慮使用反向代理的方式(例如IIS或Nginx),這些不是本文討論的內容。

  本文重點展示如何通過StaticFileMiddleware中間件,提高網站的性能。雖然這不是唯一緩存文件的方式,我們還可以通過ResponseCacheAttribute特性為ASP.NET Core Mvc的Controller和Action進行緩存的設置。

二、StaticFileMiddleware

  1.文件服務與預設緩存規則

  當創建一個ASP.NET Core的項目時,查看Startup.Configure方法,就會看到預設模板生成的添加StaticFileMiddleware中間件的方法。

public void Configure(IApplicationBuilder app)  
{
    // looging and exception handler removed for clarity

    app.UseStaticFiles();

    app.UseMvc(routes =>
    {
        routes.MapRoute(
            name: "default",
            template: "{controller=Home}/{action=Index}/{id?}");
    });
}

  這樣就使你的應用程式能夠處理,程式目錄下wwwroot目錄的靜態文件內容。在我們添加文件緩存之前,我們先要看一下StaticFileMiddleware預設的策略是怎麼樣的。當第一次載入程式時,瀏覽器將打開頁面並下載所有的資源連接。假如頁面沒有錯誤返回都是正確那麼就是返迴文件數據和Http Status為200 -OK的狀態。

 

  然後我們看下這個Http請求對應的Response Header,這裡會包含ETagLast-Modified兩個值。HTTP內容如下:

HTTP/1.1 200 OK  
Date: Sat, 15 Oct 2016 14:15:52 GMT  
Content-Type: image/svg+xml  
Last-Modified: Sat, 15 Oct 2016 13:43:34 GMT  
Accept-Ranges: bytes  
ETag: "1d226ea1f827703"  
Server: Kestrel  

  如果再次請求這個地址的話,瀏覽器將發送ETagLast-Modified的值到服務端,如果兩個值沒有變化,那麼服務端會發送304狀態到瀏覽器,那麼瀏覽器將使用之前的資源而不是重新下載一份。

   這樣就提高了,瀏覽器的響應性能,因為文件都緩存到了客戶端,並且通過304狀態,服務端與瀏覽器的請求流量得以減少。

  2.設置文件緩存時間

  當然我們都知道如果要設置某一請求的緩存,只需要設置Header為Cache-Control的值。那麼在StaticFileMiddleware中間件中,我們怎麼設置這個Header呢?

using Microsoft.Net.Http.Headers;

app.UseStaticFiles(new StaticFileOptions  
{
    OnPrepareResponse = ctx =>
    {
        const int durationInSeconds = 60 * 60 * 24;
        ctx.Context.Response.Headers[HeaderNames.CacheControl] =
            "public,max-age=" + durationInSeconds;
    }
});

 

  設置後每一個靜態文件的請求都會執行這個方法,包括200和304狀態的請求;而且在這個例子里瀏覽器會自動緩存這些文件24小時,但是在此期間並不會返回404狀態

  一旦max-age設置的時間過期,瀏覽器就不會再使用本地緩存,而去直接請求伺服器端。這樣已經避免了一些額外的請求到伺服器端了。如果我們在瀏覽器與伺服器中間使用CDN緩存文件數據的話,這樣就算客戶端瀏覽器的緩存過期了,但是請求也不會到我們的伺服器上,而是請求到CDN緩存伺服器。

  下麵我們看看文件緩存在ASP.NET Core中是如何判斷緩存失效的?.NET Core開源的代碼為我們提供了瞭解它的入口【代碼 Source Code】。核心代碼如下:

_length = _fileInfo.Length;
DateTimeOffset last = _fileInfo.LastModified;  
// Truncate to the second.
_lastModified = new DateTimeOffset(last.Year, last.Month, last.Day, last.Hour, last.Minute, last.Second, last.Offset).ToUniversalTime();
long etagHash = _lastModified.ToFileTime() ^ _length;  
_etag = new EntityTagHeaderValue('\"' + Convert.ToString(etagHash, 16) + '\"'); 

  伺服器端如果檢測到文件改變就會返回200狀態給瀏覽器,如果沒有變化則返回304狀態給瀏覽器端。

  不幸的是,一旦我們添加了緩存,瀏覽器將不再向伺服器發出請求。該文件可能已經完全改變或已被完全刪除,但如果瀏覽器不要求,伺服器將不能通知瀏覽器重新發起無緩存的請求!

  3.為靜態文件提供版本號

  通常我們都使用形如https://localhost/js/site.js?v=1 這樣的地址來解決緩存的問題。通過給靜態文件生成唯一的版本號,做為QueryString進行請求時,伺服器將重新輸出文件內容。

  在ASP.NET Core中Tag Hepers為我們提供了這樣的API:

<script src="~/js/site.js" asp-append-version="true"></script> 

  這段代碼最終在瀏覽器端會被渲染為如下Html代碼:

<script src="/js/site.js?v=Ynfdc1vuMNOWZfqTj4N3SPcebazoGXiIPgtfE-b2TO4"></script> 

  如果靜態文件發生改變,Tag Helper就是重新計算文件的哈希值,它採用 SHA256的哈希值。當然以前伺服器緩存的文件版本也隨之失效了。這個asp-append-version Tag Helper可以支持Img、Script和Link元素。

  ASP.NET Core的源代碼我們來看看是怎麼計算文件變化的:【源代碼 Source Code】 

 三、ASP.NET Core與CDN?

  我們在使用CDN時,因為還要進行開發任務,一般我們都要有兩套地址,一套是CDN上的文件地址,一套是本地調試開發用的地址。ASP.NET Core中也為我們提供了Tag Helper來解決這樣的問題。直接上代碼實例吧:

<link rel="stylesheet" href="//ajax.aspnetcdn.com/ajax/bootstrap/3.0.0/css/bootstrap.min.css"
      asp-fallback-href="~/lib/bootstrap/css/bootstrap.min.css"
      asp-fallback-test-class="hidden" 
      asp-fallback-test-property="visibility" 
      asp-fallback-test-value="hidden" />

<script src="//ajax.aspnetcdn.com/ajax/bootstrap/3.0.0/bootstrap.min.js"
        asp-fallback-src="~/lib/bootstrap/js/bootstrap.min.js"
        asp-fallback-test="window.jQuery">
</script>

  Tag Helper:asp-fallback-* 解決開發時使用的文件地址問題。 當然它也可以asp-append-version 兩個Tag Helper一起使用,這樣就實現了,在CDN文件緩存的同步問題。

四、寫在最後

  新的ASP.NET Core為我們提供了很多現有互聯網行業的解決方案,也給.NET開發人員引入了先進思想。

 

  GitHub:https://github.com/maxzhang1985/YOYOFx  如果覺還可以請Star下, 歡迎一起交流。

  

  .NET Core 和 YOYOFx 的交流群: 214741894  

參考頁面:http://qingqingquege.cnblogs.com/p/5933752.html


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • RedHat靜態Ip地址配置 依次修改以下三個文件: /etc/sysconfig/network /etc/sysconfig/network-scripts/ifcfg-eth0 /etc/resolv.conf 1./etc/sysconfig/network NETWORKING=yes N ...
  • samba服務的安裝及配置:可以在Linux中建議共用目錄給網路中的其他主機 一、安裝: sudo apt-get install samba 二、配置: 1、創建一個需要共用的目錄,並修改許可權: xx@ubuntu:~$ mkdir linux-tq2440 xx@ubuntu:~$ sudo c ...
  • 在這裡對nginx的安裝簡單的做個記錄,後續有時間的話在詳細補充。 1.yum安裝gcc gcc-c++: 2.下載必需的依賴庫:zlib 、openssl 、pcre 3.安裝依賴庫:zlib 、openssl 、pcre 4.修改配置: 添加以下內容,註意位置: 5.測試配置問是否正確: 出現以 ...
  • 1 系統調用的作用 系統調用是操作系統提供給用戶(應用程式)的一組介面,每個系統調用都有一個對應的系統調用函數來完成相應的工作。用戶通過這個介面向操作系統申請服務,如訪問硬體,管理進程等等。 應用程式和文件系統的介面是系統調用。 我們經常看到的比如fork、open、write 等等函數實際上並不是 ...
  • 首先,來查看下系統當前都開放了什麼埠,怎樣查看呢?調出cmd命令行程式,輸入命令”netstat -na“,可以看到。 接著,可以發現當前系統開放了135、445以及5357埠,而且從狀態看都處於監聽狀態”Listening“。 然後,確認自己的系統已經開放了445埠之後,我們開始著手關閉這個 ...
  • 1 安裝tftp-server sudo apt-get install tftpd-hpa sudo apt-get install tftp-hpa(如果不需要客戶端可以不安裝) 2 配置tftp伺服器 第1步: 修改/etc/default/tftpd-hpa配置文件。 將/etc/defau ...
  • 最近有在學習會話共用的配置,其中一種呢是 nginx+redis+tomcat 的會話共用配置,在記錄此會話共用配置之前呢先記錄下redis等的安裝。這篇先簡單記錄下redis的安裝,是其中一種方式,其他的後續會進行補充或單獨記錄。 準備: 解壓tcl.tar.gz,並編譯安裝: 解壓redis.t ...
  • 以前簡單介紹過web api 的設計,但是還是有很多朋友問我,如何合理的設計和實現web api。比如,介面安全,異常處理,統一數據返回等問題。所以有必要系統的總結總結 web api 的設計和實現。由於前面已經介紹過web api 的參數和返回格式的設計,《Web API系列(一)設計經驗與總結》 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...