需要改造代码,但是能解决no-cache也解决不了的问题
关于前端网页更新的问题,网上的解决方法多是在header的script里的url加入ver、配置nginx的no-cache header等。这些方法解决不了用户已经打开网页,正在使用过时网页的情况。 本文的方法可以在发布新版前端后,立即影响到已经打开的所有网页。
简单、粗暴,但是好用
这里给出了一个非常简单的方式,通过抓取网页中特定的“版本”信息,在每次发起ajax请求时,将版本信息附加到header中。然后后端的springboot接口校验此“版本”信息是否与服务端发布的index.html中的“版本”是否一致,如果不一致则抛出异常。通过控制无法使用后端接口,提示用户刷新网页的方式,实现了强制用户刷新网页的目的。
前端改造方法
在axios的interceptors.request.use中抓取网页的“版本”信息,并设置到header中,下面这个例子是vue3工程中,每次vite build生成的index.html文件的header中第一个script标签的内容。例如其中一个版本的index.html的内容如下:
<!DOCTYPE html>
<html lang="">
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width,initial-scale=1.0">
<title>SDP</title>
<script type="module" crossorigin src="/assets/index.bccd869b.js"></script>
如果每次改动代码后使用vite build,每次第一个script标签的内容都会变成新的hash值,所以通过以下js代码即可设置“版本”信息:
try {
let html = document.documentElement.outerHTML.split(' src="/assets/index.')
if (html.length > 1) {
config.headers.clientversion= html[1].split('.js"></script')[0]
}
}catch(ex) {
console.error('fail', ex)
}
vue2或其他前端技术,应该都可以通过某种方式获得每次build的唯一“版本”信息。
后端改造方法
在aop的请求切片处理中,校验HttpServletRequest的header,检查其clientversion是否存在于服务端发布的index.html文件中:
String clientVersion = httpServletRequest.getHeader("clientversion");
if (StringUtils.isEmpty(clientVersion)) {
throw new Exception("当前网页版本与服务端不一致,请刷新");
}
File f = new File(clientWebPath + "/index.html");
if (!f.exists()) {
throw new Exception("index.html文件不存在");
}
String content = FileUtil.readFile(f);
if (content.indexOf(clientVersion) < 0) {
throw new Exception("当前网页版本与服务端不一致,请刷新");
}
|