http协议 - 浅谈ETag

咱们都知道,HTTP/1.1中有一个Etag,用来判断请求的文件是否被修改。
为何要使用Etag呢?Etag主要为了解决Last-Modified没法解决的一些问题
一、一些文件也许会周期性的更改,可是他的内容并不改变(仅仅改变的修改时间),这个时候咱们并不但愿客户端认为这个文件被修改了,而从新GET;
二、某些文件修改很是频繁,好比在秒如下的时间内进行修改,(比方说1s内修改了N次),If-Modified-Since能检查到的粒度是s级的,这种修改没法判断(或者说UNIX记录MTIME只能精确到秒)
三、某些服务器不能精确的获得文件的最后修改时间;

为此,HTTP/1.1引入了Etag(Entity Tags).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段式)

说明
这里说的都是Apache 2.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,比方说设置成FileEtag Size等,就能够忽略MTime形成的Last-Modified时间修改从而影响了If-Modified-Since(IMS)这个校验了。这点和 弱Etag无关。

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

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

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

解决办法:生成Etag,由于Etag能够综合Inode,MTime和Size,能够避免这个问题
相关文章
相关标签/搜索