利用 link rel="prefetch":如何让用户的页面秒开?

大家好,我又来了😁。

我们做性能优化,通常盯着的是 首屏加载速度(FCP, LCP)。我们压缩图片、拆分代码(Code Splitting)、上CDN... 拼了老命把首屏时间从 2s 压到了 1s。

但是,用户用起来还是觉得不够快🤔。

为什么?因为当用户点击 下一页 或者 查看详情 时,那个讨厌的 Loading转圈圈 又出现了。哪怕只有 300ms,那也是一种打断

今天,我想聊一个被很多人低估的小技巧------prefetch(预获取)

我不改一行业务代码,不重构架构,只用几行代码,就让用户的下一个页面实现了秒开


preload vs prefetch

很多同学知道preload,但经常把它和prefetch搞混。这俩虽然长得像,但作用完全不同:

  • preload (预加载)我现在的页面马上就要用!

    • 优先级:High。
    • 场景:当前页面的关键字体、首屏的核心CSS/JS。
    • 如果你不用,当前页面会渲染阻塞或闪动。
  • prefetch (预获取)我现在闲着也是闲着,帮我把下一个页面的东西先下载下来呗?

    • 优先级:Low (浏览器会在网络空闲(Idle)时才去下载)。
    • 场景下一个页面可能需要的 JS Chunk、CSS 或图片。
    • 即使用户没去下一个页面,也不影响当前页面的体验,顶多浪费一点点流量。

我们现在聊的,核心就是利用prefetch,利用用户思考、滑动鼠标的那些垃圾时间,偷偷把下一个页面的资源加载好。


让详情页瞬间出现

假设我们有一个商品列表页 ,用户很有可能会点击第一个商品进入详情页

在 React/Vue 的单页应用(SPA)中,进入详情页通常意味着要加载一个Detail.chunk.js。如果等用户点击了再加载,那必然有几百毫秒的白屏。

我的做法是这样的:

当检测到用户的鼠标移动到 某个商品卡片上(或者滚动到可视区域时),我就猜:嘿,这哥们可能要点进去了🤣。

这时候,我立刻在<head>里动态插入一个prefetch标签:

JavaScript 复制代码
// 预获取下一个页面的 JS 资源
const prefetcher = (url) => {
  const link = document.createElement('link');
  link.rel = 'prefetch';
  link.href = url;
  document.head.appendChild(link);
};

// 模拟:鼠标移入商品卡片时触发
// 在 Webpack/Vite 项目中,你通常能通过 manifest 找到对应路由的 chunk 地址
card.addEventListener('mouseenter', () => {
  prefetcher('/assets/ProductDetail.23a9c.js');
  prefetcher('/assets/ProductDetail.css');
});

发生了什么?

  1. 用户鼠标悬停(hover)需要 200ms~500ms 才会点击。
  2. 浏览器利用这几百毫秒的空闲,已经在后台把ProductDetail.js下载好了 ,并放进了HTTP 缓存(Disk Cache)
  3. 当用户真的点击时,路由跳转,请求资源 -> 浏览器发现缓存里有 -> 直接从磁盘读取 -> 耗时 0ms!

用户感觉就是:手指刚落下,页面就砰地一下出来了。 这种爽快感,是任何代码压缩都给不了的。


手动写mouseenter太累了,而且我怎么知道打包后的文件名叫啥?

Google Chrome 团队早就想到了,他们开源了一个神器叫 quicklink

它的原理极其简单且暴力:

检测视口(Viewport)内的链接,自动给它们加上prefetch。

JavaScript 复制代码
// 1. 安装
// npm install quicklink

// 2. 在你的页面加载完成后调用
import { listen } from 'quicklink';

listen({
  // 选项:只在网络环境好的时候预取 (默认就是这样)
  // 选项:限制预取的数量,防止浪费流量
  limit: 5 
});

只要这一行代码!

当用户滚动列表时,进入视口的那些商品链接对应的 JS/CSS 资源,就会被自动静默下载。用户点哪个,哪个就是秒开。

Next.js 的 <Link> 组件,默认就内置了类似的功能。 这也是为什么 Next.js 应用通常感觉比普通 React 应用快的原因之一。


prefetch 也不是万能的

虽然它很爽,但作为一个负责任的工程师,我必须告诉你几个坑😖:

  1. 必须配合正确的缓存策略 (Cache-Control)

prefetch 下载的资源是放在 HTTP 缓存里的。如果你的服务器设置了Cache-Control: no-store,那用户点击时,浏览器还是会重新发请求,你的预获取就白做了。确保你的静态资源有长缓存(Max-Age)策略。

  1. 别预取 API 接口 (慎用😭)

有些人想连 API 数据也预取。这很危险。因为数据的时效性很难保证。你预取了库存为1,用户点进去时可能已经卖光了。除非是静态博客类的内容,否则只建议预取 JS/CSS/图片 等静态资源。

  1. 移动端省流模式

在移动端,流量就是钱。浏览器通常比较智能,在省流模式或弱网下,会忽略prefetch指令。我们在做手动实现时,也可以通过 navigator.connection.saveData 来判断,如果是省流模式,就别预取了,做个好人😂。


性能优化,不仅仅是把资源做小 ,更是对用户行为的一种预判

link rel="prefetch" ,当你还在苦哈哈地抠那 10kb 的打包体积时,不妨试试这个技巧,说不定你的网站会飞起来。

有时候,让用户感觉快,比真的快,更重要。

相关推荐
IT_陈寒1 小时前
Vite的热更新突然不香了,排查三小时差点砸键盘
前端·人工智能·后端
子兮曰2 小时前
Agency-Agents 深度解析:400+ AI 专家的"梦之队"如何重塑开发工作流
前端·后端·vibecoding
山河木马2 小时前
渲染管线-计算得到gl_Position(顶点着色器)之后续GPU流程
javascript·webgl·图形学
竹林8182 小时前
用 The Graph 查询链上数据实战:从手搓 RPC 到 Subgraph,我的 NFT 项目数据加载快了 10 倍
前端·javascript
妙码生花3 小时前
从 PHP 到 AI + Golang,程序员自救转型手记(十九):点选验证码代码逐行目检
前端·后端·go
Awu12273 小时前
⚡从零开发 Agent CLI(五)实现一个可治理、可扩展的工具系统
前端·人工智能·claude
咪库咪库咪4 小时前
Vue3-生命周期
前端
莪_幻尘4 小时前
你的 AI Skill 越多越蠢?Token 上下文爆炸的求生指南
前端·ai编程
lichenyang4535 小时前
从 has.echo 到异步 API 注册表:一次 ASCF API 回调不触发的排查复盘
前端
林瞅瞅5 小时前
Nuxt3 项目部署 Nginx 防盗链后特定 JS 文件 403 问题修复方案
前端