ASP.NET Core中的Http缓存

ASP.NET Core中的Http缓存

Http响应缓存可减少客户端或代理对 web服务器发出的请求数。响应缓存还减少了 web服务器生成响应所需的工作量。响应缓存由 Http请求中的 header控制。

而 ASP.NETCore对其都有相应的实现,并不需要了解里面的工作细节,即可对其进行良好的控制。

了解HTTP缓存

Http协议中定义了许多缓存,但总体可以分为强缓存协商缓存两类。

强缓存

强缓存是指缓存命中时,客户端不会向服务器发请求,浏览器 F12能看到响应状态码为 200, size为 fromcache,它的实现有以下几种方式:

Expires - 绝对时间

示例: Expires:Thu,31Dec203723:59:59GMT,就表示缓存有效期至2037年12月31日,在这之前浏览器不会向服务器发请求了(除非按 F5Ctrl+F5刷新)。

Cache-Control - 相对时间/更多控制

绝对时间是一个绝对时间,因为计算时不方便;而且服务端是依据服务器的时间来返回,但客户端却需要依据客户的时间来判断,因此也容易失去控制。

Cache-Control有以下选项(可以多选):

  1. max-age: 指定一个时间长度,在这个时间段内缓存是有效的,单位是秒( s)。

    例如设置 Cache-Control:max-age=31536000,也就是说缓存有效期为 31536000/24/60/60=365天。

  2. s-maxage: 同 max-age,覆盖 max-age、 Expires,但仅适用于共享缓存,在私有缓存中被忽略。

  3. public: 表明响应可以被任何对象(发送请求的客户端、代理服务器等等)缓存。

  4. private: 表明响应只能被单个用户(可能是操作系统用户、浏览器用户)缓存,是非共享的,不能被代理服务器缓存。

  5. no-cache: 强制所有缓存了该响应的用户,在使用已缓存的数据前,发送带验证器的请求到服务器。(不是字面意思上的不缓存

  6. no-store: 禁止缓存,每次请求都要向服务器重新获取数据。

  7. must-revalidate: 指定如果页面是过期的,则去服务器进行获取。(意思是浏览器在某些情况下,缓存失效后仍可使用老缓存,加了这个头,失效后就必须验证,并不是字面上有没有过期都验证

其中最有意思的要数 no-cache和 must-revalidate了,因为它们的表现都不是字面意义。

no-cache并不是字面上的不缓存,而是会一直服务端验证(真实意义很像字面上的 must-revalidate)。

而 must-revalidate是只是为了给浏览器强调,缓存过期后,千万要遵守约定重新验证。

协商缓存

协商缓存是指缓存命中时,服务器返回 Http状态码为 304但无内容( Body),没命中时返回 200有内容。

在要精细控制时,协商缓存比强缓存更有用,它有 Last-Modified和 ETag两种。

Last-Modified/If-Modify-Since(对比修改时间)

示例:

服务器:Last-Modified: Sat, 27 Jun 2015 16:48:38 GMT
客户端:If-Modified-Since: Sat, 27 Jun 2015 16:48:38 GMT

ETag/If-None-Match(对比校验码)

服务器:ETag: W/"0a0b8e05663d11:0"
客户端:If-None-Match: W/"0a0b8e05663d11:0"

清缓存要点

  • 按 F5刷新时,强缓存失效

  • 按 Ctrl+F5刷新时 强缓存和协商缓存都失效

ASP.NET Core的Http缓存

ASP.NETCore中提供了 ResponseCacheAttribute来实现缓存,它的定义如下:

[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, AllowMultiple = false, Inherited = true)]
public class ResponseCacheAttribute : Attribute, IFilterFactory, IFilterMetadata, IOrderedFilter
{public ResponseCacheAttribute();public string CacheProfileName { get; set; }public int Duration { get; set; }public bool IsReusable { get; }public ResponseCacheLocation Location { get; set; }public bool NoStore { get; set; }public int Order { get; set; }public string VaryByHeader { get; set; }public string[] VaryByQueryKeys { get; set; }
}

其中, ResponseCacheLocation定义了缓存的位置,是重点:

//     Determines the value for the "Cache-control" header in the response.
public enum ResponseCacheLocation
{//     Cached in both proxies and client. Sets "Cache-control" header to "public".Any = 0,//     Cached only in the client. Sets "Cache-control" header to "private".Client = 1,//     "Cache-control" and "Pragma" headers are set to "no-cache".None = 2
}

注意看源文件中的注释, Any表示 Cache-Control:public, Client表示 Cache-Control:private, None表示 Cache-Control:no-cache

注意 ResponseCacheLocation并没有定义将缓存放到服务器的选项。

其中 Duration表示缓存时间,单位为秒,它将翻译为 max-age

另外可以通过 VaryByHeader和 VaryByQueryKeys来配置缓存要不要通过 header和 querystring来变化,其中 VaryByHeader是通过 Http协议中的 Vary头来实现的, VaryByQueryKeys必须通过 Middleware来实现。

不要误会,所有 ResponseCacheAttribute的属性配置都不会在服务端缓存你的响应数据(虽然你可能有这种错觉),它和输出缓存不同,它没有状态,只用来做客户端强缓存。

如果不想缓存,则设置 NoStore=true,它会设置 cache-control:no-store,我们知道 no-store的真实意思是不缓存。一般还会同时设置 Location=ResponseCacheLocation.None,它会设置 cache-control:no-cache(真实意思是表示一定会验证)。

注意单独设置 Location=ResponseCacheLocation.None而不设置 NoStore并不会有任何效果。

示例1

这是一个很典型的使用示例:

public class HomeController : Controller
{[ResponseCache(Duration = 3600, Location = ResponseCacheLocation.Client)]public IActionResult Data(){return Json(DateTime.Now);}
}

我定义了 3600秒的缓存,并且 cache-control应该为 private,生成的 Http缓存头可以通过如下 C#代码来验证:

using var http = new HttpClient();
var resp1 = await http.GetAsync("https://localhost:55555/home/data");
Console.WriteLine(resp1.Headers.CacheControl.ToString());
Console.WriteLine(await resp1.Content.ReadAsStringAsync());

输入结果如下:

max-age=3600, private
"2020-03-07T21:35:01.5843686+08:00"

另外, ResponseCacheAttribute也可以定义在 Controller级别上,表示整个 Controller都受到缓存的影响。

CacheProfileName示例

另外,如果需要共用缓存配置,可以使用 CacheProfileName,将缓存提前定义好,之后直接传入这个定义名即可使用:

.ConfigureServices(s =>
{s.AddControllers(o =>{o.CacheProfiles.Add("3500", new CacheProfile{Duration = 3500, Location = ResponseCacheLocation.Client,});});
});

这样我就定义了一个名为 3500的缓存,稍后在 Controller中我只需传入 CacheProfileName=3500即可:

public class HomeController : Controller
{[ResponseCache(CacheProfileName = "3500")]public IActionResult Data(){return Json(DateTime.Now);}
}

总结

Http缓存分为强缓存和协商缓存, ASP.NETCore提供了便利的 ResponseCacheAttribute实现了强缓存,还能通过 Profile来批量配置多个缓存点。

但 ASP.NET MVC并没有提供协商缓存实现,因为这些多半和业务逻辑相关,需要自己写代码。静态文件是特例, Microsoft.AspNetCore.StaticFiles中提供有,因为静态文件的逻辑很清晰。

ASP.NET中的 OutputCacheAttribute在 ASP.NETCore中不复存在,取而代之的是 app/services.AddResponseCaching(),这些和 Http协议不相关。

有机会我会具体聊聊这些缓存。

喜欢的朋友请关注我的微信公众号:【DotNet骚操作】

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/311256.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SQL(五) - 表的创建以及操作

创建表 建表语句的语法格式: create table 表名(字段名1 数据类型,字段名2 数据类型,字段名3 数据类型,....);MySql常用数据类型 BLOB 二进制大对象(存储图片、视频等流媒体信息) Binary Large OBject (对应java中的Object&…

Istio 2020 年 Roadmap——一切为了商用

原文地址:https://preliminary.istio.io/zh/blog/2020/tradewinds-2020/,由 ServiceMesher 社区翻译。Istio 解决了人们在运行微服务时遇到的实际问题。甚至早期的预发行版本就已经可以帮助用户诊断其体系架构中的延迟,提高服务的可靠性以及透…

SQL(七) - 事务、索引、视图

事务(Transaction) 3.1、什么是事务? 一个事务是一个完整的业务逻辑单元,不可再分。 比如:银行账户转账,从A账户向B账户转账10000.需要执行两条update语句: update t_act set balance balan…

如何编写高性能的C#代码(二)

使用Benchmark.NET对C# 代码进行基准测试的简介在我以前的文章中[10],我介绍了该系列文章[11],在其中我将分享我的经验,同时了解C#和.NET Core(corefx)框架的新性能。在本文中,我想着重于对现有…

如何编写高性能的C#代码(一)

原文来自互联网,由长沙DotNET技术社区编译。如译文侵犯您的署名权或版权,请联系小编,小编将在24小时内删除。作者介绍:史蒂夫戈登(Steve Gordon)是Microsoft MVP,Pluralsight的作者,…

从Java转向.NET/C#,Are You OK?

最近由于项目变动,需要用.NET/C#做开发,经过一段时间的学习和培训,对这个技术栈有了一定的理解。大家可能都知道Java和.NET/C#很像,这里粗略的把两者做一个对比,希望对感兴趣的童鞋有所帮助。如果现在有人问我&#xf…

树的节点值之和

题目背景 墨家家主有棵树。 题目描述 给定一个保存树节点信息的数据结构,它包含了树节点唯一的 id ,树节点值 和 直系子节点的 id 。 比如,树节点1是树节点2的父节点,树节点2是树节点3的父节点。他们相应的树节点值为 9 , 4 , …

.NET Core开发实战(第21课:中间件:掌控请求处理过程的关键)--学习笔记(上)...

21 | 中间件:掌控请求处理过程的关键这一节讲解一下如何通过中间件来管理请求处理过程中间件工作原理next 表示后面有一个委托,每一层每一层套下去可以在任意的中间件来决定在后面的中间件之前执行什么,或者说在所有中间件执行完之后执行什么…

简单的二叉树创建与遍历

编一个程序,读入用户输入的一串先序遍历字符串,根据此字符串建立一个二叉树(以指针方式存储)。 例如如下的先序遍历字符串: ABC##DE#G##F### 其中“#”表示的是空格,空格字符代表空树。建立起此二叉树以后&…

疫情期间,千万级系统宕机N次,老板撂下狠话:没法把性提升10倍,全员解雇!...

性能调优整体思路作为一名团队技术核心,如何让系统跑得通、跑得稳、跑得快是必然会面对的场景。性能分析是一个大课题,不同的架构、不同的应用场景、不同的程序语言分析的方法若有差异,抽象一下大致分为两类:自底向上:…

Anaconda创建python虚拟环境

在创建虚拟环境之前首先我们需要打开命令终端:Win R 输入cmd 或者直接打开Anaconda Prompt(Anaconda) pycharm下载历史版本地址:https://www.jetbrains.com/pycharm/download/other.html Anaconda下载历史版本地址:ht…

[蓝桥杯][算法提高VIP]夺宝奇兵-递推+记忆化搜索

题目描述 在一座山上,有很多很多珠宝,它们散落在山底通往山顶的每条道路上,不同道路上的珠宝的数目也各不相同.下图为一张藏宝地图: 7 3 8 8 1 0 2 7 4 4 4 5 2 6 5 ”夺宝奇兵”从山下出发,到达山顶,如何选路才能得到最多的珠宝呢?在上图所示例子中,按照5-> 7-> 8-&g…

梯度下降与线性回归

对于代价函数: loss∑i(y^−yi)2loss\sum_i{(\hat{y}-y_i)}^2loss∑i​(y^​−yi​)2 loss∑i(w∗xib−yi)2loss\sum_i{(w*x_ib-y_i)}^2loss∑i​(w∗xi​b−yi​)2 最常见的代价函数:均方差代价函数(Mean-Square Error,MSE&…

.NET Core开发实战(第21课:中间件:掌控请求处理过程的关键)--学习笔记(下)...

21 | 中间件:掌控请求处理过程的关键如果在 Map 的时候逻辑复杂一点,不仅仅判断它的 URL 地址,而且要做特殊的判断的话,可以这么做把判断逻辑变成一个委托我们要判断当我们的请求地址包含 abc 的时候,输出 new abcapp.…

英伟达3060Ti安装GPU版本TensorFlow2.X和Pytorch

查看Python与TensorFlow对应版本 安装GPU版本的TensorFlow的时候,我们需要考虑的一个问题是Python版本与TensorFlow版本的对应关系,可以参考下面这个链接: Python对应TensorFlow CPU版本 GPU版本 查看显卡驱动对应的CUDA版本并且下载安装 …

200行代码,7个对象——让你了解ASP.NET Core框架的本质[3.x版]

2019年1月19日,微软技术(苏州)俱乐部成立,我受邀在成立大会上作了一个名为《ASP.NET Core框架揭秘》的分享。在此次分享中,我按照ASP.NET Core自身的运行原理和设计思想创建了一个 “迷你版” 的ASP.NET Core框架&…

逻辑回归(二)

逻辑回归 在学习逻辑回归之前我们先回顾一下线性回归。线性回归解决的是回归问题,简单来说就是,我们需要找到一个函数,这个函数需要尽可能的拟合所有训练集的样本点。 逻辑回归解决的是分类问题,它的目标是找到一个函数&#x…

上元节的灯会(灭)-区间dp

题目背景 上元节的庙会上,牛宝靠自己的聪明才智成功破解了花灯阵,点亮了在场所有花灯,但他没料到的是这个游戏包含AB两个项目,A项目就是点亮所有花灯,而B项目则是熄灭所有花灯。不过点亮的是花灯阵,熄灭的…

Asp.Net Core 中IdentityServer4 授权中心之应用实战

一、前言查阅了大多数相关资料,搜索到的IdentityServer4 的应用文章大多是比较简单并且多是翻译官网的文档编写的,我这里在 Asp.Net Core 中IdentityServer4 的应用分析中会以一个电商系统架构升级过程中普遍会遇到的场景进行实战性讲述分析,…

交通标志识别项目教程

项目结构图 下载好项目压缩包后解压,得到以上的文件,首先将画红圈的文件删除(如果有) 安装软件 安装Anaconda 安装Pycharm 安装格式工厂 在上图中这个位置输入cmd回车,即可打开命令终端。用这样的方式打开命令终端…