前端缓存踩坑指南:如何优雅地解决浏览器缓存问题?
浏览器缓存,配置得当,它能让页面飞起来;配置错了,一次小小的上线,就能把你扔进线上 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 更新事件,给用户提示“有新版本啦,点我刷新”。
三、前端缓存怎么设计才靠谱?
我们总结一个简化但实用的资源缓存模型:
资源类型 | 缓存策略 | 原因 |
|
| 确保每次加载最新入口 |
JS/CSS(打包产物) |
| 文件名唯一,放一年都没问题 |
图片 / 字体 | CDN 强缓存 | 资源大且变化少 |
接口数据 |
/ 加时间戳 | 实时性要求高 |
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 = () => {// 显示刷新提示
}
六、调试缓存的几种方法
- Chrome DevTools → Network 里看资源的
Status
是不是(from disk cache)
或(from memory cache)
- 查看
Response Headers
是否有Cache-Control
、ETag
curl -I
直接看 HTTP 响应头- Service Worker 状态 → Chrome DevTools → Application → Service Worker
总结
浏览器缓存细节极多,尤其是前端更新策略和 CDN/服务器协作时,要明确方向:
- 明确哪些资源该缓存、缓存多久
- 更新机制清晰、可控、不怕上线
- 构建流程自动化,不靠人肉维护
相关文章:
前端缓存踩坑指南:如何优雅地解决浏览器缓存问题?
浏览器缓存,配置得当,它能让页面飞起来;配置错了,一次小小的上线,就能把你扔进线上 bug 的坑里。你可能遇到过这些情况: 部署上线了,结果用户还在加载旧的 JS;接口数据改了…...

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…...
从软件到硬件:三大主流架构的特点与优劣详解
常见的架构包括软件架构、企业架构、硬件架构等,以下是对这几种常见架构的分析: 一、软件架构 1.分层架构 描述:分层架构是一种经典的软件架构模式,将软件系统按照功能划分为不同的层次,一般包括表现层(…...
STM32printf重定向到串口含armcc和gcc两种方案
STM32串口重定向:MDK与GCC环境下需重写的函数差异 在嵌入式开发中,尤其是使用 STM32系列微控制器 的项目中,调试信息的输出是不可或缺的一部分。为了方便调试,开发者通常会选择将 printf 等标准输出函数通过 UART 串口发送到 PC …...

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

Webug4.0靶场通关笔记11- 第15关任意文件下载与第16关MySQL配置文件下载
目录 一、文件下载 二、第15关 任意文件下载 1.打开靶场 2.源码分析 3.渗透实战 三、第16关 MySQL配置文件下载 1.打开靶场 2.源码分析 3.渗透实战 (1)Windows系统 (2)Linux系统 四、渗透防御 一、文件下载 本文通过…...
【中间件】brpc_基础_remote_task_queue
文章目录 remote task queue1 简介2 核心功能2.1 任务提交与分发2.2 无锁或低锁设计2.3 与 bthread 深度集成2.4 流量控制与背压 3 关键实现机制3.1 数据结构3.2 任务提交接口3.3 任务窃取(Work Stealing)3.4 同步与唤醒 4 性能优化5 典型应用场景6 代码…...
maven坐标导入jar包时剔除不需要的内容
maven坐标导入jar包时剔除不需要的内容 问题描述解决方案 问题描述 maven坐标导入jar包时剔除不需要的内容 解决方案 Spring Boot 默认使用 Logback,需在 pom.xml 中排除其依赖: <dependency><groupId>org.springframework.boot</gro…...

k8s监控方案实践(一):部署Prometheus与Node Exporter
k8s监控方案实践(一):部署Prometheus与Node Exporter 文章目录 k8s监控方案实践(一):部署Prometheus与Node Exporter一、Prometheus简介二、PrometheusNode Exporter实战部署1. 创建Namespace(p…...
ValueError: Could not find common ancestor of[]
ValueError: Could not find common ancestor of [0004_deadstockstathistorymodel, 0026_remove_orderdetailmodel_order_no]说明 Django 当前在 尝试生成迁移或者执行迁移 时,发现你的迁移历史“断裂”了: 你这个 App 的迁移历史有两个分支,…...

具身系列——比较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 解码器实现(示例…...
PyTorch_自动微分模块
自动微分 (Autograd) 模块对张量做了进一步的封装,具有自动求导功能。自动微分模块是构成神经网络训练的必要模块,在神经网络的反向传播过程中,Autograd 模块基于正向计算的结果对当前的参数进行微分计算,从而实现网络权重参数的更…...
【Git】【commit】查看未推送的提交查看指定commit的修改内容合并不连续的commit
文章目录 1. 查看未推送的提交方法一 :git status方法二:git log方法三:git cherry方法四:git rev-list 2. 查看指定commit的修改方法一:git show方法二:git log方法三:git diff 3. 合并不连续的…...
手写 Vue 源码 === 依赖清理机制详解
目录 引言 响应式系统基础回顾 依赖清理的必要性 ReactiveEffect 类的设计 依赖清理的三个关键函数 1. preCleanEffect:执行前的准备 2. trackEffects:依赖收集与 diff 算法 3. postCleanEffect:执行后的清理 4. cleanDepEffect:清理依赖 实际案例分析 依赖清理算…...

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…...
DDR在PCB布局布线时的注意事项及设计要点
一、布局注意事项 控制器与DDR颗粒的布局 靠近原则:控制器与DDR颗粒应尽量靠近,缩短时钟(CLK)、地址/控制线(CA)、数据线(DQ/DQS)的走线长度,减少信号延迟差异。 分组隔…...
【每天学习一点点】使用Python的pathlib模块分割文件路径
使用Python的pathlib模块分割文件路径 pathlib模块(Python 3.4)提供了面向对象的文件系统路径操作方式,比传统的os.path更加直观和易用。以下是使用pathlib分割文件路径的几种方法: 基本路径分割 from pathlib import Path# 创…...
Hydra详细教程:入门、入狱,和使用与注意事项
警告:本文档仅供学习和授权测试目的使用。未经授权对计算机系统进行渗透测试是非法行为。请务必在获得明确许可的情况下使用Hydra,并遵守所有适用的法律法规。滥用此工具可能导致严重的法律后果。 什么是Hydra? Hydra是一款非常流行的开源网…...
【C++游戏引擎开发】第32篇:物理引擎(Bullet)—约束系统
一、约束系统基础理论 1.1 物理约束的本质 1.1.1 约束的数学描述 在刚体动力学中,约束的本质是通过数学方程限制刚体的运动自由度。对于两个刚体A和B的约束关系,可以用以下方程表示: Φ ( q A , q B , t ) = 0...

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

理解 Elasticsearch 的评分机制和 Explain API
作者:来自 Elastic Kofi Bartlett 深入了解 Elasticsearch 的评分机制并探索 Explain API。 想获得 Elastic 认证吗?查看下一期 Elasticsearch Engineer 培训的时间! Elasticsearch 拥有大量新功能,帮助你为你的使用场景构建最佳…...
NGINX `ngx_http_charset_module` 字符集声明与编码转换
一、模块定位与功能 ngx_http_charset_module 主要提供两大能力: 响应头声明:在 Content-Type 头部自动添加 ; charsetXXX,告知客户端所用字符集。单向编码转换:在 NGINX 层将一种单字节编码(如 koi8-r、windows-125…...

视频编解码学习三之显示器
整理自:显示器_百度百科,触摸屏_百度百科,百度安全验证 分为阴极射线管显示器(CRT),等离子显示器PDP,液晶显示器LCD 液晶显示器的组成。一般来说,液晶显示器由以下几个部分组成: […...
Python中的re库详细用法与代码解析
目录 1. 前言 2. 正则表达式的基本概念 2.1 什么是正则表达式? 2.2 常用元字符 3. re库的适应场景 3.1 验证用户输入 3.2 从文本中提取信息 3.3 文本替换与格式化 3.4 分割复杂字符串 3.5 数据清洗与预处理 4. re库的核心功能详解 4.1 re.match()&#…...

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

13.Excel:分列
一 分列的作用 将一个单元格中的内容拆分到两个或多个单元格当中。 二 如何使用 1.常规分列使用 注意:分列功能一次只能拆分一列。 长度一致或者数据间有分隔符。 补充:快速选择一列。 CTRL shift 向下箭头。 补充:中英文逗号不同。 可以先通…...
第十六届蓝桥杯大赛软件赛C/C++大学B组部分题解
第十六届蓝桥杯大赛软件赛C/C大学B组题解 试题A: 移动距离 问题描述 小明初始在二维平面的原点,他想前往坐标(233,666)。在移动过程中,他只能采用以下两种移动方式,并且这两种移动方式可以交替、不限次数地使用: 水平向右移动…...

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