eTag

eTag 其实是一个页面的一个HASH值,当浏览器刷新页面时,浏览器会发送页面的ETAG值,让服务器进行比较,服务器如果发现eTag值没变,则返回304,浏览器则从缓存中获取页面,如果不相等,则重新获取页面,如果eTag过期,则重新获取页面。

 

我们都知道,HTTP/1.1中有一个Etag,用来判断请求的文件是否被修改。
为什么要使用Etag呢?Etag主要为了解决Last-Modified无法解决的一些问题


1、一些文件也许会周期性的更改,但是他的内容并不改变(仅仅改变的修改时间),这个时候我们并不希望客户端认为这个文件被修改了,而重新GET;
2、某些文件修改非常频繁,比如在秒以下的时间内进行修改,(比方说1s内修改了N次),If-Modified-Since能检查到的粒度是s级的,这种修改无法判断(或者说UNIX记录MTIME只能精确到秒)
3、某些服务器不能精确的得到文件的最后修改时间;

为此,HTTP/1.1引入了Etag(EntityTags).Etag仅仅是一个和文件相关的标记,可以是一个版本标记,比如说v1.0.0或者说"2e681a-6-5d044840"这么一串看起来很神秘的编码。但是HTTP/1.1 标准并没有规定Etag的内容是什么或者说要怎么实现,唯一规定的是Etag需要放在""内。

Etag由服务器端生成,客户端通过If-Match或者说If-None-Match这个条件判断请求来验证资源是否修改。我们常见的是使用If-None-Match.请求一个文件的流程可能如下:


====第一次请求===
1.客户端发起HTTP GET请求一个文件;
2.服务器处理请求,返回文件内容和一堆Header,当然包括Etag(例如"2e681a-6-5d044840")(假设服务器支持Etag生成和已经开启了Etag). 状态码200

====第二次请求===
1.客户端发起HTTP GET请求一个文件,注意这个时候客户端同时发送一个If-None-Match头,这个头的内容就是我们第一次请求时服务器返回的Etag:2e681a-6-5d044840
2.服务器判断发送过来的Etag和计算出来的Etag匹配,因此If-None-Match为False,不返回200,返回304,客户端继续使用本地缓存;

流程很简单,问题是,如果服务器又设置了Cache-Control:max-age和Expires呢,怎么办?
答案是同时使用,也就是说在完全匹配If-Modified-Since和If-None-Match即检查完修改时间和Etag之后,服务器才能返回304.(不要陷入到底使用谁的问题怪圈)

我们来看Apache中的Etag实现。
1.Apache首先判断是不是弱Etag,这个留在下面讲。如果不是,进入第二种情况:

强Etag根据配置文件中的配置来设置Etag值,默认的Apache的FileEtag设置为:
FileEtag INode Mtime Size
也就是根据这三个属性来生成Etag值,他们之间通过一些算法来实现,并输出成hex的格式,相邻属性之间用-分隔,比如:
Etag "2e681a-6-5d044840"
这里面的三个段,分别代表了INode,MTime,Size根据算法算出的值的Hex格式,(如果你在这里看到了非Hex里面的字符(也就是0-f),那你可能看见神了:))

当然,我们可以改变Apache的FileEtag设置,比如设置成FileEtag Size,那么得到的Etag可能为:
Etag "6"
总之,设置了几个段,Etag值就有几个段。(不要误以为Etag就是固定的3段式)

说明
这里说的都是Apache2.2里面的Etag实现,因为HTTP/1.1并没有规定Etag必须是什么样的实现或者格式,因此,你也可以修改或者完全编写自己的算法得到Etag,比如 "2e681a65d044840",客户端会记住并缓存下这个Etag(Windows里面保存在哪里,我还没找到:(),下次访问的时候直接拿这个值去和服务器生成的Etag对比。

注意
不管怎么样的算法,在服务器端都要进行计算,计算就有开销,会带来性能损失。因此为了榨干这一点点性能,不少网站完全把Etag禁用了(比如Yahoo!),这其实不符合HTTP/1.1的规定,因为HTTP/1.1总是鼓励服务器尽可能的开启Etag。

弱校验(弱Etag)
重新考虑前面提到的3个问题:
问题1、一些文件也许会周期性的更改,但是他的内容并不改变(仅仅改变的修改时间),这个时候我们并不希望客户端认为这个文件被修改了,而重新GET;

解决办法:如果使用强Etag,每次得会要求重新GET页面,如果使用Etag,比方说设置成FileEtagSize等,就可以忽略MTime造成的Last-Modified时间修改从而影响了If-Modified-Since(IMS)这个校验了。这点和弱Etag无关。

问题2、某些文件修改非常频繁,比如在秒以下的时间内进行修改,(比方说1s内修改了N次),If-Modified-Since能检查到的粒度是s级的,这种修改无法判断(或者说UNIX记录MTIME只能精确到秒)

解决办法:如果是这种情况,Apache会自动判断请求时间和修改时间之间的差值,如果小于1s,Apache会认为这个文件在这1秒内可能会再次被修改,因此生成一个弱Etag(WeakEtag),这个Etag仅仅基于MTime来生成,因此MTime只能精确到s,所以1s内生成的Etag总是一样,这样就避免了使用强Etag造成的1s内频繁的刷新Cache的情况。(貌似不用Etag,仅仅使用Last-Modified就可以解决,但是这针对的仅仅是修改超级频繁的情况,很多文件可能同时也使用强Etag验证)。弱Etag以W/开始,比如:W/"2e681a"

问题3、某些服务器不能精确的得到文件的最后修改时间;

 

Cache-control常见取值:  (服务器控制的快取策略)

private

no-cache

max-age

must-revalidate

 

 


### ETag 技术原理 ETag(实体标签)是 HTTP/1.1 协议中定义的一种响应头字段,用于提供资源的特定版本标识符。ETag 通常由服务器生成,是一个字符串,代表资源在特定时间点的状态。当客户端再次请求相同资源时,可以使用 ETag 来验证缓存的有效性。 ETag 的主要作用是优化缓存验证机制。客户端在后续请求中通过 `If-None-Match` 请求头发送先前获取的 ETag 值。服务器收到请求后,会将当前资源的 ETag 与客户端提供的值进行比较。如果两者相同,说明资源未发生变化,服务器返回 `304 Not Modified` 状态码;如果不同,则返回新的资源内容和状态码 `200 OK`。 ETag 的生成方式通常基于资源的某些属性,例如文件的最后修改时间、大小或内容的哈希值。例如,服务器可以使用文件名和最后修改时间来生成 ETag: ```csharp ETag eTag = new ETag(fileInfo.Name + fileInfo.LastWriteTimeUtc.ToString()); ``` 这种方式确保了当文件内容发生变化时,ETag 也会随之更新,从而触发客户端重新下载资源[^3]。 ### 使用场景 ETag 主要用于以下场景: - **缓存验证**:当客户端缓存了某个资源时,可以通过 ETag 来验证缓存是否仍然有效。这减少了不必要的数据传输,提高了性能。 - **减少服务器负载**:通过缓存验证机制,服务器可以避免重复发送相同的资源内容,从而降低带宽消耗和服务器负载。 - **提高用户体验**:由于减少了不必要的资源下载,用户的浏览器可以更快地加载页面,提升了整体的用户体验。 ### HTTP 头信息 ETag 是 HTTP 响应头的一部分,常见的相关头信息包括: - **ETag**: 服务器生成的资源标识符。 - **If-None-Match**: 客户端在请求中携带的 ETag 值,用于验证缓存的有效性。 在实际应用中,ETag 通常与其他缓存控制机制(如 `Cache-Control` 和 `Last-Modified`)结合使用,以实现更高效的缓存策略。例如,`Cache-Control` 可以指定缓存的最大年龄,而 `Last-Modified` 提供了资源的最后修改时间,进一步增强了缓存验证的准确性[^2]。 ### 示例代码 以下是一个简单的示例,展示了如何在 C# 中实现 ETag 的基本功能: ```csharp public class ETag : IHeader { private string value; public ETag(string value) { Value = value; WebOperationContext.Current.OutgoingResponse.ETag = Value; } public string Value { get { return value; } set { this.value = value; } } public void AddHTTPHeader(ResponseContext context) { context.WriteHttpHeader(Value); } } ``` 在这个示例中,`ETag` 类实现了 `IHeader` 接口,并通过构造函数接收一个字符串参数来初始化 ETag 的值。该值随后被设置为响应头中的 ETag 字段。 ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值