前端缓存踩坑指南:如何优雅地解决浏览器缓存问题?

news2025/5/10 0:59:43

浏览器缓存,配置得当,它能让页面飞起来;配置错了,一次小小的上线,就能把你扔进线上 bug 的坑里。你可能遇到过这些情况:

  • 部署上线了,结果用户还在加载旧的 JS;
  • 接口数据改了,页面却还拿着几分钟前的老版本;
  • Service Worker 缓了一大堆东西,结果连新页面都加载不出来……

一、浏览器缓存到底是怎么一回事?

别被“缓存”两个字骗了,它不只是简单地“存一下资源”。

浏览器的缓存机制其实是分层的,而且各层的策略和触发条件都不一样:

1.1 三类缓存存储层

缓存类型

生命周期

典型用途

Memory Cache

页面打开期间,关闭即清

JS/CSS 快速复用

Disk Cache

浏览器层面,长期有效

HTML、图片等

Service Worker Cache

自定义、可离线

PWA 场景或定制缓存策略

1.2 两种核心缓存逻辑

浏览器判断一个资源能不能用缓存,基本分这两种逻辑:

✅ 强缓存(直接用,不发请求)

通过 HTTP 响应头:

Cache-Control: max-age=31536000

或者:

Expires: Wed, 21 Oct 2025 07:28:00 GMT

只要浏览器还在有效期内,连请求都不发,直接从缓存取资源。

✅ 协商缓存(发请求,看是否要更新)

如果没命中强缓存,浏览器会尝试和服务器协商资源是否变了:

If-None-Match: "e1d3f...etag"

如果服务器返回 304,说明资源没变,浏览器用本地缓存;否则重新拉。


二、常见“缓存坑”还原现场

🧨 场景一:JS 改了,用户却还是老页面

怎么踩的?

前端用了默认打包配置,JS 文件名不变,浏览器命中了强缓存,用户根本没加载到新版资源。

怎么救?

  • 打包输出文件名加上 hash,比如:
app.3e9fbd.js

Vite/Webpack 都支持 [contenthash]

  • 确保 HTML 不缓存!因为 HTML 决定了要加载哪个 JS 文件,如果 HTML 没更新,JS 永远也更新不了。
Cache-Control: no-store
  • 如果用了 CDN,还要记得清缓存或者上版本号。

🧨 场景二:接口数据老是卡住,怎么刷新都不变?

怎么踩的?

很多时候是 GET 请求被浏览器或者中间代理缓存了。

比如:

GET /api/user/info

但服务端没写清楚 Cache-Control,浏览器自动缓存了响应。

怎么救?

  • 后端返回头加上:
Cache-Control: no-cache, no-store, must-revalidate
  • 前端请求的时候,给 URL 加时间戳,强制打破缓存:
fetch(`/api/user/info?_t=${Date.now()}`)

🧨 场景三:Service Worker 缓了个寂寞,更新根本不生效

怎么踩的?

你可能用了 Workbox 或手写了 SW 缓静态资源,但是没处理好版本更新流程,结果用户永远在旧版本里打转。

怎么救?

  • 每次构建产出时更新 SW 文件里的版本号:
const CACHE_NAME = 'my-app-v2.0.1';
  • install 阶段调用 self.skipWaiting(),跳过等待状态。
  • activate 阶段用 clients.claim() 接管所有页面。
  • 页面上监听 SW 更新事件,给用户提示“有新版本啦,点我刷新”。

三、前端缓存怎么设计才靠谱?

我们总结一个简化但实用的资源缓存模型:

资源类型

缓存策略

原因

index.html

no-store

确保每次加载最新入口

JS/CSS(打包产物)

max-age=1y, immutable

文件名唯一,放一年都没问题

图片 / 字体

CDN 强缓存

资源大且变化少

接口数据

no-cache

/ 加时间戳

实时性要求高

PWA 离线资源

SW 控制

手动缓存 + 可更新


四、工程化落地建议

别只靠脑子记,我们得把缓存策略写进构建流程里:

4.1 打包配置文件指纹

Webpack/Vite:

output: {
  filename: '[name].[contenthash].js'
}

4.2 HTML 注入正确的资源引用

Webpack 用 HtmlWebpackPlugin;Vite 自动处理。

4.3 CDN 清缓存脚本

比如:

npx aliyun-cdn-purge path/to/index.html

或者自动脚本 + Git Hook 联动清缓存。


五、Service Worker

建议不要上来就用 SW 缓全站资源,除非你非常清楚它的行为。

常用策略:

  • 图片、静态文件:Cache First
  • 接口数据:Network First
  • HTML 入口:Network Only
  • 通知/消息等:Network Only + fallback

搭配 Workbox,配置如下:

workbox.routing.registerRoute(
  /\.(?:png|jpg|jpeg|svg)$/,
  new workbox.strategies.CacheFirst({
    cacheName: 'images',
  })
);

更新版本时,只需要更新缓存名,比如:

const CACHE_NAME = 'v2.0.3';

页面监听更新:

navigator.serviceWorker.onupdatefound = () => {
  // 显示刷新提示
}

六、调试缓存的几种方法

  1. Chrome DevTools → Network 里看资源的 Status 是不是 (from disk cache)(from memory cache)
  2. 查看 Response Headers 是否有 Cache-ControlETag
  3. curl -I 直接看 HTTP 响应头
  4. Service Worker 状态 → Chrome DevTools → Application → Service Worker

总结

浏览器缓存细节极多,尤其是前端更新策略和 CDN/服务器协作时,要明确方向:

  • 明确哪些资源该缓存、缓存多久
  • 更新机制清晰、可控、不怕上线
  • 构建流程自动化,不靠人肉维护

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2371916.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Ubuntu 单机多卡部署脚本: vLLM + DeepSeek 70B

# 部署脚本:Ubuntu vLLM DeepSeek 70B # 执行前请确保:1. 系统为 Ubuntu 20.04/22.04 2. 拥有NVIDIA显卡(显存≥24G) # 保存两个文件 1 init.sh 初始化 2、test.sh 测试 # init.sh #!/bin/bash # 系统更新与基础依赖sudo apt update && s…

为了摸鱼和吃瓜,我开发了一个网站

平时上班真的比较累,摸鱼和吃瓜还要跳转多个平台的话,就累上加累了。 所以做了一个聚合了全网主流平台热搜的网站。 目前市面上确实有很多这种网站了,所以目前最主要有两点和他们不同: 给热搜列表增加了配图,刷的时候…

Webug4.0靶场通关笔记11- 第15关任意文件下载与第16关MySQL配置文件下载

目录 一、文件下载 二、第15关 任意文件下载 1.打开靶场 2.源码分析 3.渗透实战 三、第16关 MySQL配置文件下载 1.打开靶场 2.源码分析 3.渗透实战 (1)Windows系统 (2)Linux系统 四、渗透防御 一、文件下载 本文通过…

k8s监控方案实践(一):部署Prometheus与Node Exporter

k8s监控方案实践(一):部署Prometheus与Node Exporter 文章目录 k8s监控方案实践(一):部署Prometheus与Node Exporter一、Prometheus简介二、PrometheusNode Exporter实战部署1. 创建Namespace(p…

具身系列——比较3种vpg算法方式玩CartPole游戏(强化学习)

文档1方式参考:https://gitee.com/chencib/ailib/blob/master/rl/vpg_baseline_cartpole.py 文档2方式参考:https://gitee.com/chencib/ailib/blob/master/rl/vpg_batchupdate_cartpole.py 文档3方式参考:https://gitee.com/chencib/ailib/bl…

面向未来的 TCP 协议设计:可扩展与兼容并存

目录 1.设计思路 (1)完整数据结构(字节布局) 1)字段解释: 2)Flags字段设计(1字节位图) (2)进阶版 Java 解码器实现(示例&#xf…

LSB图像信息隐藏系统(MATLAB)

图像信息隐藏系统 系统概述 图像信息隐藏系统是一个基于MATLAB开发的图像隐写工具,采用自适应LSB(最低有效位)隐写算法,实现了信息在图像中的隐藏与提取功能。系统配备了直观的图形用户界面,支持图像分析、信息隐藏、…

C++GO语言微服务项目之 go语言基础语法

目录 01 变量定义 02 自增语法 03 指针 04 go不支持的语法 05 string 06 定长数组-forrange 07 动态数组追加元素 08 切片截取-copy-make介绍 09 map介绍 10 函数 11 内存逃逸 12 import 13 命令行参数-switch 14 标签与continue-goto-break配合使用 15 枚举cons…

最新字节跳动运维云原生面经分享

继续分享最新的go面经。 今天分享的是组织内部的朋友在字节的go运维工程师岗位的云原生方向的面经,涉及Prometheus、Kubernetes、CI/CD、网络代理、MySQL主从、Redis哨兵、系统调优及基础命令行工具等知识点,问题我都整理在下面了 面经详解 Prometheus …

理解 Elasticsearch 的评分机制和 Explain API

作者:来自 Elastic Kofi Bartlett 深入了解 Elasticsearch 的评分机制并探索 Explain API。 想获得 Elastic 认证吗?查看下一期 Elasticsearch Engineer 培训的时间! Elasticsearch 拥有大量新功能,帮助你为你的使用场景构建最佳…

视频编解码学习三之显示器

整理自:显示器_百度百科,触摸屏_百度百科,百度安全验证 分为阴极射线管显示器(CRT),等离子显示器PDP,液晶显示器LCD 液晶显示器的组成。一般来说,液晶显示器由以下几个部分组成: […

K8s网络从0到1

K8s网络从0到1 前言 K8s是一个强大的平台,但它的网络比较复杂,涉及很多概念,例如Pod网络,Service网络,Cluster IPs,NodePort,LoadBalancer和Ingress等等。为了帮助大家理解,模仿TC…

13.Excel:分列

一 分列的作用 将一个单元格中的内容拆分到两个或多个单元格当中。 二 如何使用 1.常规分列使用 注意:分列功能一次只能拆分一列。 长度一致或者数据间有分隔符。 补充:快速选择一列。 CTRL shift 向下箭头。 补充:中英文逗号不同。 可以先通…

计算机网络应用层(5)-- P2P文件分发视频流和内容分发网

💓个人主页:mooridy 💓专栏地址:《计算机网络:自顶向下方法》 大纲式阅读笔记_mooridy的博客-CSDN博客 💓本博客内容为《计算机网络:自顶向下方法》第二章应用层第五、六节知识梳理 关注我&…

Gin优雅关闭 graceful-shutdown

文章目录 优雅关闭示例 - Close 方法项目结构使用方法代码如下代码说明如果去掉代码中的数字1,会发生什么 优雅关闭示例项目结构使用方法使用上下文通知不使用上下文通知 代码 notify-without-context-server.go代码说明 代码 notify-with-context-server.go代码说明…

五子棋html

<!DOCTYPE html> <html lang"zh-CN"> <head> <meta charset"UTF-8" /> <meta name"viewport" content"widthdevice-width, initial-scale1" /> <title>五子棋游戏</title> <style>bo…

JavaWeb:后端web基础(TomcatServletHTTP)

一、今日内容 二、Tomcat 介绍与使用 介绍 基本使用 小结 配置 配置 查找进程 三、Servlet 什么是Servlet 快速入门 需求 步骤 1.新建工程-模块&#xff08;Maven&#xff09; 2.修改打包方式-war 3.编写代码 /*** 可以选择继承HttpServlet*/ WebServlet("/hello&q…

缓存(1):三级缓存

三级缓存是指什么 我们常说的三级缓存如下&#xff1a; CPU三级缓存Spring三级缓存应用架构&#xff08;JVM、分布式缓存、db&#xff09;三级缓存 CPU 基本概念 CPU 的访问速度每 18 个月就会翻 倍&#xff0c;相当于每年增⻓ 60% 左右&#xff0c;内存的速度当然也会不断…

Cursor —— AI编辑器 使用详解

Cursor - The AI Code Editor 一、Cursor 是什么&#xff1f; Cursor 是一款优秀的AI代码编辑器&#xff0c;它内置了 Deepseek-R1、GPT-4、Claude等 AI 模型。 简单说&#xff0c;就是&#xff1a;Cursor VS Code 编辑器 AI 大模型 Cursor 功能特性&#xff08;代码补全、…

Pytorch-CUDA版本环境配置

Pytorch-CUDA版本环境配置 电脑如果是Windows平台下的Nvidia GPU的用户&#xff0c;需配置Pytorch的CUDA版本&#xff0c;分为三步&#xff1a; 1. 安装或更新NVIDA显卡驱动 官方驱动下载地址&#xff1a; https://www.nvidia.cn/Download/index.aspx?langcn 2. 安装CUDA Too…