纯前端实现更新检测

通过判断打包后的html文件中的js入口是否发生变化,进而实现前端的代码更新

为了使打包后的文件带有hash值,需要对vite打包进行配置

javascript 复制代码
import { defineConfig } from 'vite';
import vue from '@vitejs/plugin-vue';
import { resolve } from 'path';
import AutoImport from 'unplugin-auto-import/vite'
import Components from 'unplugin-vue-components/vite'
import { ElementPlusResolver } from 'unplugin-vue-components/resolvers'

export default defineConfig({
  base: './',
  plugins: [
    vue(), AutoImport({
      resolvers: [ElementPlusResolver()],
    }),
    Components({
      resolvers: [ElementPlusResolver()],
    }),],
  envDir: './', // .env所在目录
  build: {
    target: 'es2015',
    outDir: 'dist',
    rollupOptions: {
      output: {
        manualChunks(id) { 
          if (id.includes('node_modules')) {
            return id.toString().split('node_modules/')[1].split('/')[0].toString();
          }
        },
        // 文件附带上hash
        entryFileNames: '[name]-[hash].js',
        chunkFileNames: '[name]-[hash].js',
        assetFileNames: '[name]-[hash].[ext]'
      }
    },
    minify: 'esbuild', 
  },
  resolve: {
    alias: {
      "@": resolve(__dirname, 'src'),
    },
    extensions: ['.vue', '.js']
  },
  server: {
    proxy: {
      '/api': {
        target: 'http://localhost:8083',
        changeOrigin: true,
        rewrite: (path) => path.replace(/^\/api/, "")
      }
    }
  },
});

update.js的主要功能是定期检查网页中的脚本标签是否有更新,并在检测到新版本时提示用户刷新页面以确保平台正常使用。

javascript 复制代码
// update.js
import { ElMessageBox } from 'element-plus';

let timer = undefined;
const url = import.meta.env.VITE_UPDATE_URL;

function cmpSets(set1, set2) {
    if (set1.size !== set2.size) return false;
    for (let item of set1) {
        if (!set2.has(item)) return false;
    }
    return true;
}
// 更新提示框
function updateNotice() {
    ElMessageBox({
        title: '更新提示!',
        message: "检测到新版本,请立即刷新以确保平台正常使用",
        confirmButtonText: '确定',
        type: 'warning',
    }).finally(() => {
        window.location.reload();
    });
}

// 获取页面中的脚本标签src属性的哈希值集合,忽略查询参数
async function getSrcHash() {
    try {
        const html = await fetch(url).then((res) => res.text());
        const scriptSrcRegex = /<script\b[^>]*src="([^"]*)"/gi;
        const scriptSrcs = [...html.matchAll(scriptSrcRegex)].map(match => match[1]);
        // 开发环境中入口文件包含保存时的时间戳,判断时移除此参数以免开发时一直提示更新
        const cleanSrcs = scriptSrcs.map(src => src.split('?')[0]); 
        const encodedSrcs = new Set(cleanSrcs.map(src => encodeURIComponent(src)));
        return encodedSrcs;
    } catch (error) {
        console.error('Failed to fetch script hashes:', error);
        return new Set();
    }
}

// 比较当前脚本标签哈希值与新获取的哈希值
async function cmpHash() {
    try {
        const newHash = await getSrcHash();
        const storedHash = JSON.parse(localStorage.getItem('curHash')) || [];

        // 如果是新用户或首次访问,直接保存哈希值并退出函数
        if (storedHash.length === 0) {
            localStorage.setItem('curHash', JSON.stringify([...newHash]));
            return;
        }
        // 合并新旧哈希值集合
        let curHash = new Set(storedHash);
        if (!cmpSets(curHash, newHash)) {
            console.info("new:", newHash);
            console.info("old:", curHash);
            console.log("更新提示")
            clearInterval(timer);
            updateNotice();
        }

        // 保存最新的哈希值集合到localStorage
        localStorage.setItem('curHash', JSON.stringify([...newHash]));
    } catch (error) {
        console.error('Error comparing script hashes:', error);
    }
}


// 设置定时器,定期检查脚本更新
timer = setInterval(cmpHash, 30 * 1000);

// 页面加载时比较哈希值
async function init() {
    await cmpHash();
}
// 页面加载时执行初始化函数
init();

最后在main.js入口文件引入即可

javascript 复制代码
import { createApp } from 'vue'
import App from './App.vue'
import ElementPlus from 'element-plus'
import 'element-plus/dist/index.css'
import router from './router/index.ts'
import "./update.js" //引入自动更新脚本
import { createPinia } from 'pinia'

const pinia = createPinia()
const app = createApp(App)
app.use(router)
app.use(pinia)
app.use(ElementPlus)
app.mount('#app')

通过篡改localstorage中的curHash,可引导更新,效果如下

30s更新一次算频繁吗?

相关推荐
2401_857439691 小时前
SSM 架构下 Vue 电脑测评系统:为电脑性能评估赋能
开发语言·php
迷雾漫步者1 小时前
Flutter组件————FloatingActionButton
前端·flutter·dart
SoraLuna1 小时前
「Mac畅玩鸿蒙与硬件47」UI互动应用篇24 - 虚拟音乐控制台
开发语言·macos·ui·华为·harmonyos
向前看-1 小时前
验证码机制
前端·后端
xlsw_1 小时前
java全栈day20--Web后端实战(Mybatis基础2)
java·开发语言·mybatis
燃先生._.2 小时前
Day-03 Vue(生命周期、生命周期钩子八个函数、工程化开发和脚手架、组件化开发、根组件、局部注册和全局注册的步骤)
前端·javascript·vue.js
Dream_Snowar2 小时前
速通Python 第三节
开发语言·python
高山我梦口香糖3 小时前
[react]searchParams转普通对象
开发语言·前端·javascript
m0_748235243 小时前
前端实现获取后端返回的文件流并下载
前端·状态模式