需要改造代码,但是能解决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("当前网页版本与服务端不一致,请刷新");
    }
Logo

前往低代码交流专区

更多推荐