.NET 6開發(fā)之實現(xiàn)緩存過程詳解
需求
有的時候為了減少客戶端請求相同資源的邏輯重復執(zhí)行,我們會考慮使用一些緩存的方式,在.NET 6中,我們可以借助框架提供的中間件來實現(xiàn)請求資源的緩存。
目標
實現(xiàn)請求結(jié)果的緩存。
原理與思路
對于在.NET6中實現(xiàn)緩存,我們可以使用響應緩存中間件ResponseCaching
來實現(xiàn),同時可以使用Marvin.Cache.Headers
來為我們提供更多的緩存相關的屬性。
實現(xiàn)
使用原生ResponseCaching實現(xiàn)緩存
既然是中間件,我們便在Program中引入:
Program.cs
// 省略其他... // 配置緩存中間件 builder.Services.AddResponseCaching(); builder.Services.AddControllers(); // ... // 使用緩存中間件 app.UseResponseCaching(); app.MapControllers();
在使用方法上,有幾種方式可以實現(xiàn)配置:1)進行全局的配置,應用于所有添加了相同ProfileName
的ResponseCache
的Controller響應;2)對單個Controller/Action進行配置,應用于當前作用的Controller/Action;3)全局配置后,由單個Controller/Action覆蓋全局配置。我們會演示1)和3)的場景。
我們準備使用獲取所有TodoLists的接口進行演示。
先看如何進行全局配置:
Program.cs
// 省略其他... builder.Services.AddControllers(options => { options.CacheProfiles.Add("60SecondDuration", new CacheProfile { Duration = 60 }); });
驗證1: 全局配置Caching
首先給我們要進行驗證的Action添加屬性:
TodoListController.cs
// 省略其他... [HttpGet] [ResponseCache(CacheProfileName = "60SecondDuration")] public async Task<ApiResponse<List<TodoListBriefDto>>> Get() { return ApiResponse<List<TodoListBriefDto>>.Success(await _mediator.Send(new GetTodosQuery())); }
啟動Api
項目,第一次執(zhí)行獲取TodoLists
的請求:
請求
響應
響應頭中多了一個cache-control
字段用于指明緩存的類型(public)以及過期時間為60s:
如果你是使用Postman
或者Insomia發(fā)送的請求,那么在過期前再次發(fā)起相同請求的返回頭中會再多出一個Age
字段,用于表明該資源當前緩存了多少秒(Hoppscotch
我沒找到可以在哪里設置,所以下面的截圖是來自Insomia
,如果有哪位老哥知道的可以教一下):
同時如果觀察日志的話會發(fā)現(xiàn),第二次請求并沒有實際執(zhí)行SQL語句,這也證明了第二次請求的返回來自緩存:
如果間隔60s以上我們再去發(fā)送相同的請求,會發(fā)現(xiàn)日志中是這樣的:
可以看到緩存已經(jīng)失效了,此時需要重新向數(shù)據(jù)庫查詢返回數(shù)據(jù),并將這次請求結(jié)果緩存起來。
驗證2: 單個Action覆蓋全局配置
我們還是使用這個接口,但是修改一下屬性:
TodoListController.cs
[HttpGet] [ResponseCache(Duration = 120)] public async Task<ApiResponse<List<TodoListBriefDto>>> Get() { return ApiResponse<List<TodoListBriefDto>>.Success(await _mediator.Send(new GetTodosQuery())); }
重新啟動Api
項目,第一次執(zhí)行獲取TodoLists
的請求,請求和驗證1相同,我們來看響應中的變化:
響應
可以看到失效時間已經(jīng)變?yōu)?20s了,其他不再一一驗證。
使用Marvin.Cache.Headers實現(xiàn)更多緩存功能
在緩存中還有一個問題是,如果判斷緩存的數(shù)據(jù)內(nèi)容已經(jīng)變化,就需要去獲取最新的響應而不是直接從緩存中取值。這是借助緩存校驗來完成的,而常使用的方式是通過Etag
實現(xiàn)。示意的過程如下:
如果首次請求資源,API會在響應頭中添加Etag
和Last-Modified
字段:
當客戶端再次請求資源時,由于緩存自身是不知道資源有沒有被修改,所以緩存會攜帶If-None-Match
字段(和客戶端收到的Etag
值相等)和If-Modified-Since
字段(和客戶端收到的Last-Modified
值相等)到API端,如果校驗發(fā)現(xiàn)資源沒有發(fā)生修改,那么API端無需重新獲取資源,直接返回304
字段(NotModifed)給緩存,緩存給客戶端返回值。如果校驗發(fā)現(xiàn)資源發(fā)生了修改,那么API將會返回新的結(jié)果。
我們給Api
項目添加Nuget包Marvin.Cache.Headers
,來實現(xiàn)此功能。
首先向Program
中添加服務以及引入中間件:
Program.cs
builder.Services.AddResponseCaching(); builder.Services.AddHttpCacheHeaders( expirationOptions => { expirationOptions.MaxAge = 180; expirationOptions.CacheLocation = CacheLocation.Private; }, validateOptions => { validateOptions.MustRevalidate = true; }); // 省略其他... app.UseResponseCaching(); app.UseHttpCacheHeaders();
同時我們需要移除之前添加的ResponseCache
屬性,因為新引入的庫已經(jīng)幫我們完成了,當然我們也可以通過以下方式覆蓋全局配置:
[HttpCacheExpiration(CacheLocation = CacheLocation.Public, MaxAge = 60)] [HttpCacheValidation(MustRevalidate = false)]
覆蓋規(guī)則和框架內(nèi)置的規(guī)則是一致的,我不會繼續(xù)演示。
驗證3: 緩存校驗
請求仍然是獲取所有的TodoLists
:
響應
我們暫時只關注響應頭:
如果在緩存失效前我們添加了一個新的TodoList
,在請求頭中添加If-None-Match=53154EEFAE230D733827DBDE49B42AF9
再執(zhí)行獲取請求:
可以看到在失效時間到期之內(nèi),Etag
值已經(jīng)發(fā)生了變化,校驗表明資源已經(jīng)改變,需要重新獲取。
如果我們再次獲取相同的資源,會得到304
返回:
一點擴展
但是如果我們仔細觀察和思考就會發(fā)現(xiàn),框架在實現(xiàn)緩存校驗上存在兩個問題:
If-None-Match
頭字段是我們手動添加模擬的,這本應該由緩存中間件來完成;- 在響應
304
的情況下,實際上是沒有返回響應體的,即緩存中未修改的資源沒有返回;
這兩個問題是由框架內(nèi)建的ResponseCaching
庫導致的,可以認為它沒有正確地實現(xiàn)緩存校驗機制。為此我們有一些替代方案可供參考:
當然使用專門的CDN
來做緩存也是可以的。
總結(jié)
在本文中我們主要演示了如何借助框架的緩存機制來實現(xiàn)請求資源的緩存,盡管在緩存校驗的實現(xiàn)上,官方提供的庫目前來看并沒有能很好地完成功能以外,對于我們基本的使用場景來說已經(jīng)夠用了。下一篇文章我們來看怎么實現(xiàn)接口的限流。
參考資料
1.Varnish
3.Squid
到此這篇關于.NET 6開發(fā)之實現(xiàn)緩存過程詳解的文章就介紹到這了,更多相關.NET 6實現(xiàn)緩存內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
ASP.NET Core使用JWT認證授權(quán)的方法
這篇文章主要介紹了ASP.NET Core使用JWT認證授權(quán)的方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-11-11Linux上使用Docker部署ASP.NET?Core應用程序
這篇文章介紹了使用Docker部署ASP.NET?Core應用程序的方法,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2022-03-03ASP.NET利用MD.DLL轉(zhuǎn)EXCEL具體實現(xiàn)
首先引入MD.dll 文件(附有下載地址)然后建立無CS文件的DownExcel.aspx 文件,接下來是調(diào)用方法,感興趣的朋友可以參考下哈2013-05-05asp.net fileupload控件上傳文件與多文件上傳
這篇文章主要介紹了asp.net fileupload控件上傳文件的方法,fileupload控件多文件上傳,以及fileupload上傳時實現(xiàn)文件驗證的方法,需要的朋友可以參考下2014-11-11ASPX向ASCX傳值以及文本創(chuàng)建圖片(附源碼)
把用戶在TextBox輸入的文字創(chuàng)建為一個圖片,ASCX的ImageButton的ImageUrl重新指向這剛產(chǎn)生的圖片,接下來介紹下ASPX向ASCX傳值,感興趣的朋友可以參考下哈2013-03-03