使用 JavaScript
记录 cookie
更新浏览器 html
缓存文件。
@[toc]
1. 原理说明
1.利用浏览器响应 html 文件请求的响应头信息 last-modified
(文件最后更新时间) 和 ETag
来判断 html 文件是否更新;
2.实际使用中,因为无法获取 ETag
信息,所以实际仅仅利用了请求头的 last-modified
信息;
2. 参考文档
Web 开发技术请参见 HTTP请参见 HTTP HeadersLast-Modified:
https://developer.mozilla.org/zh-CN/docs/Web/HTTP/Headers/Last-ModifiedWeb 开发技术请参见 HTTP请参见 HTTP HeadersETag:
https://developer.mozilla.org/zh-CN/docs/Web/HTTP/Headers/ETagHTTP的请求头标签 If-Modified-Since:
https://www.cnblogs.com/zh2000g/archive/2010/03/22/1692002.html你知道 http 响应头中的 ETag 是如何生成的吗:
https://segmentfault.com/a/1190000021251877
总结:
nginx 中 etag 由响应头的 Last-Modified 与 Content-Length 表示为十六进制组合而成。
如果 http 响应头中 ETag 值改变了,是否意味着文件内容一定已经更改:
https://github.com/shfshanyue/Daily-Question/issues/113
答案是不一定,由服务器中 ETag 的生成算法决定;
比如 nginx 中的 etag 由 last_modified 与 content_length 组成,而 last_modified 又由 mtime 组成;
当编辑文件却未更改文件内容时,或者执行 touch file 的时候,mtime 也会改变,此时 etag 改变,但是文件内容没有更改。
3. 代码实现
// 本代码依赖于 jQuery ;
// jQuery(document).ready(function(){}); 意义为在 DOM 加载完毕后执行了ready()方法,简写为 jQuery(function(){});
$( document ).ready( function ( ) {
// 定义一个函数,用来记录 cookie ;
function HomePageSetCookie_For_CheckHomePageHtmlFileIsChange_For_ReloadHomePageHtmlFile ( ) {
// 判断是否为网站的首页;
if ( window.location.pathname == "/index.html" ) {
// 建立 xhr 请求;
var xhr = new XMLHttpRequest();
if( !xhr ){
xhr = new ActiveXObject("Microsoft.XMLHTTP");
}
// 参数 HEAD 需要配合 nginx 的配置文件 if ($request_method !~* GET|POST|HEAD) { return 444; } ;否则无法获取回应;
xhr.open( "HEAD" , window.location.href + "?timestamp=" + Date.now() , true ) ;
xhr.onreadystatechange=function(){
// xhr 请求成功;
if( xhr.readyState == 4 && xhr.status == 200 ){
// 定义函数,删除特殊字符串;
function DeleteTitle_SpecialString( LastModified ) {
LastModified = LastModified.replace(/\s+/g,'') ;
LastModified = ( ( LastModified.replace( /,/g , '' ) ).replace( /;/g , '' ) ).replace( /:/g , '' ) ;
return LastModified ;
}
// 提取 xhr 响应头的 last-modified ;
var ResponseHeaders_LastModified = xhr.getResponseHeader( "last-modified" ) ;
ResponseHeaders_LastModified = DeleteTitle_SpecialString( ResponseHeaders_LastModified ) ;
// 判断是否已经存在 cookie 记录,正则代码([^;]+)匹配非分号字符; (?:;|$)匹配分号或行尾 ;_HasIndexHtml 表示网站首页的 window.location.pathname 包含 "/index.html" 字符串 ;
if ( document.cookie.match(/HomePageSetCookie_ResponseHeaders_LastModified_HasIndexHtml=([^;]+)(;|$)/) != null ) {
// 读取 cookie 记录;
var Array_LastModified_HasIndexHtml = document.cookie.match(/HomePageSetCookie_ResponseHeaders_LastModified_HasIndexHtml=([^;]+)(;|$)/);
var Cookie_LastModified_HasIndexHtml = Array_LastModified_HasIndexHtml[ 0 ].substring( Array_LastModified_HasIndexHtml[ 0 ].indexOf( "=" ) + 1 , Array_LastModified_HasIndexHtml[ 0 ].length ) ;
Cookie_LastModified_HasIndexHtml = DeleteTitle_SpecialString( Cookie_LastModified_HasIndexHtml ) ;
// 根据 last-modified 判断 html 文件已经修改;
if ( Cookie_LastModified_HasIndexHtml != ( ResponseHeaders_LastModified + "_HasIndexHtml" ) ) {
// 更新 cookie 记录;
var cookie_path = "/" ;
document.cookie = "HomePageSetCookie_ResponseHeaders_LastModified_HasIndexHtml="+ResponseHeaders_LastModified+"_HasIndexHtml; expires=Fri, 31 Dec 9999 23:59:59 GMT; " + ( window.location.host == "" ? "" : "domain=" + ( ( window.location.hostname ).indexOf( "www." ) != -1 ? ( window.location.hostname ).substring( 3 , ( window.location.hostname ).length ) : "." + window.location.hostname ) ) + "; path=" + cookie_path + "; SameSite=none; Secure=true" ;
// 刷新网站的首页,更新 html 文件缓存;注意:要先更新 cookie 记录,后执行 window.location.reload( ); 否则会导致页面刷新的死循环 ;
window.location.reload( );
// 重新赋值 variable_For_MultiTabPageHtmlReLoad_HasIndexHtml 变量;
// 用于浏览器同时打开多个标签页流量网站首页的时候,更新全部标签页的 html 文件缓存;
window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = document.cookie.match(/HomePageSetCookie_ResponseHeaders_LastModified_HasIndexHtml=([^;]+)(;|$)/);
window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml[ 0 ].substring( window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml[ 0 ].indexOf( "=" ) + 1 , window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml[ 0 ].length ) ;
window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = DeleteTitle_SpecialString( window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml ) ;
}
else { // if ( Cookie_LastModified_HasIndexHtml == ( ResponseHeaders_LastModified + "_HasIndexHtml" ) ) {
// 更新 cookie 记录;
var cookie_path = "/" ;
document.cookie = "HomePageSetCookie_ResponseHeaders_LastModified_HasIndexHtml="+ResponseHeaders_LastModified+"_HasIndexHtml; expires=Fri, 31 Dec 9999 23:59:59 GMT; " + ( window.location.host == "" ? "" : "domain=" + ( ( window.location.hostname ).indexOf( "www." ) != -1 ? ( window.location.hostname ).substring( 3 , ( window.location.hostname ).length ) : "." + window.location.hostname ) ) + "; path=" + cookie_path + "; SameSite=none; Secure=true" ;
// 通过 variable_For_MultiTabPageHtmlReLoad_HasIndexHtml 变量,判断是否其他的标签页已经更新了 html 文件的同时,更新了 cookie 记录 ;
// 其他标签页虽然可以更新 cookie ,但是不能更新 本标签页的 window. 对象属性;
if ( $.type( window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml ) == "undefined" ) {
// 定义一个变量,用来解决在 两个 标签页中 同时打开首页的时候:其中一个首页更新了 html 文件,并且重写了 cookie 记录;导致另外一个首页更新 html 文件失败的问题;
// 因为第一个 首页更新 html 文件的时候,已经将 cookie 记录重写为最新的值,此时第二个标签页比对 ResponseHeaders_LastModified 与 cookie 记录,发现是相同的,就不会去更新 html 页面;
// 此处定义一个变量就是为了解决此问题,先将 旧的 cookie 记录挂载到 window. 的全局变量上;
// 如果发现 ResponseHeaders_LastModified 与 cookie 记录相同,那么就再比对最新的 cookie 记录(也就是最后的 ResponseHeaders_LastModified ) 与 全局变量 window. 上挂载的 cookie 旧记录是否相同;因为其他标签页虽然可以更新 cookie ,但是不能更新 本标签页的 window. 对象属性;
// 不相同的话,说明其他的 html 文件已经更新了缓存,那么本标签页就重新载入 html 缓存文件;
var variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = document.cookie.match(/HomePageSetCookie_ResponseHeaders_LastModified_HasIndexHtml=([^;]+)(;|$)/);
variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = variable_For_MultiTabPageHtmlReLoad_HasIndexHtml[ 0 ].substring( variable_For_MultiTabPageHtmlReLoad_HasIndexHtml[ 0 ].indexOf( "=" ) + 1 , variable_For_MultiTabPageHtmlReLoad_HasIndexHtml[ 0 ].length ) ;
variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = DeleteTitle_SpecialString( variable_For_MultiTabPageHtmlReLoad_HasIndexHtml ) ;
window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = variable_For_MultiTabPageHtmlReLoad_HasIndexHtml ;
}
else if ( window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml != ( ResponseHeaders_LastModified + "_HasIndexHtml" ) ) {
// 刷新网站的首页,更新 html 文件缓存;
window.location.reload( );
// 重新赋值 variable_For_MultiTabPageHtmlReLoad_HasIndexHtml 变量;
window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = document.cookie.match(/HomePageSetCookie_ResponseHeaders_LastModified_HasIndexHtml=([^;]+)(;|$)/);
window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml[ 0 ].substring( window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml[ 0 ].indexOf( "=" ) + 1 , window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml[ 0 ].length ) ;
window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = DeleteTitle_SpecialString( window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml ) ;
}
}
}
else { // == null ,没有 cookie 记录;
var cookie_path = "/" ;
document.cookie = "HomePageSetCookie_ResponseHeaders_LastModified_HasIndexHtml="+ResponseHeaders_LastModified+"_HasIndexHtml; expires=Fri, 31 Dec 9999 23:59:59 GMT; " + ( window.location.host == "" ? "" : "domain=" + ( ( window.location.hostname ).indexOf( "www." ) != -1 ? ( window.location.hostname ).substring( 3 , ( window.location.hostname ).length ) : "." + window.location.hostname ) ) + "; path=" + cookie_path + "; SameSite=none; Secure=true" ;
// 定义一个变量,用来解决在 两个 标签页中 同时打开首页的时候:其中一个首页更新了 html 文件,并且重写了 cookie 记录;导致另外一个首页更新 html 文件失败的问题;
// 因为第一个 首页更新 html 文件的时候,已经将 cookie 记录重写为最新的值,此时第二个标签页比对 ResponseHeaders_LastModified 与 cookie 记录,发现是相同的,就不会去更新 html 页面;
// 此处定义一个变量就是为了解决此问题,先将 旧的 cookie 记录挂载到 window. 的全局变量上;
// 如果发现 ResponseHeaders_LastModified 与 cookie 记录相同,那么就再比对最新的 cookie 记录(也就是最后的 ResponseHeaders_LastModified ) 与 全局变量 window. 上挂载的 cookie 旧记录是否相同;因为其他标签页虽然可以更新 cookie ,但是不能更新 本标签页的 window. 对象属性;
// 不相同的话,说明其他的 html 文件已经更新了缓存,那么本标签页就重新载入 html 缓存文件;
var variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = document.cookie.match(/HomePageSetCookie_ResponseHeaders_LastModified_HasIndexHtml=([^;]+)(;|$)/);
variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = variable_For_MultiTabPageHtmlReLoad_HasIndexHtml[ 0 ].substring( variable_For_MultiTabPageHtmlReLoad_HasIndexHtml[ 0 ].indexOf( "=" ) + 1 , variable_For_MultiTabPageHtmlReLoad_HasIndexHtml[ 0 ].length ) ;
variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = DeleteTitle_SpecialString( variable_For_MultiTabPageHtmlReLoad_HasIndexHtml ) ;
window.variable_For_MultiTabPageHtmlReLoad_HasIndexHtml = variable_For_MultiTabPageHtmlReLoad_HasIndexHtml ;
}
} // if( xhr.readyState == 4 && xhr.status == 200 ) 语句结束
} // xhr.onreadystatechange=function() 结束
xhr.send(null);
}
}
// 页面打开的时候,立即执行一次函数,判断是否更新 html 文件;
setTimeout ( function ( ) {
HomePageSetCookie_For_CheckHomePageHtmlFileIsChange_For_ReloadHomePageHtmlFile ( ) ;
} , 1 ) ;
// 每隔 15 分钟,定时检查 html 文件是否需要更新缓存;
setInterval ( function ( ) {
HomePageSetCookie_For_CheckHomePageHtmlFileIsChange_For_ReloadHomePageHtmlFile ( ) ;
} , 900000 ) ;
} ) ; // $(document).ready(function(){