利用 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 的打包体积时,不妨试试这个技巧,说不定你的网站会飞起来。

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

相关推荐
摘星编程10 分钟前
用React Native开发OpenHarmony应用:StickyHeader粘性标题
javascript·react native·react.js
A_nanda14 分钟前
c# 用VUE+elmentPlus生成简单管理系统
javascript·vue.js·c#
天天进步201531 分钟前
Motia事件驱动的内核:深入适配器(Adapter)层看消息队列的流转
javascript
北极糊的狐35 分钟前
若依项目vue前端启动键入npm run dev 报错:不是内部或外部命令,也不是可运行的程序或批处理文件。
前端·javascript·vue.js
XRJ040618xrj38 分钟前
Nginx下构建PC站点
服务器·前端·nginx
We་ct1 小时前
LeetCode 289. 生命游戏:题解+优化,从基础到原地最优
前端·算法·leetcode·矩阵·typescript
有诺千金1 小时前
VUE3入门很简单(4)---组件通信(props)
前端·javascript·vue.js
2501_944711431 小时前
Vue-路由懒加载与组件懒加载
前端·javascript·vue.js
雨季6662 小时前
Flutter 三端应用实战:OpenHarmony “心流之泉”——在碎片洪流中,为你筑一眼专注的清泉
开发语言·前端·flutter·交互
换日线°2 小时前
前端3D炫酷展开效果
前端·3d