📖 小说网站的预导航实战:link 预加载 + fetch + 前进后退全支持

一、前言

在小说网站里,点击「下一章」通常需要等待网络请求,体验不佳。为此,很多网站会采用 预导航 技术:提前获取下一章内容,点击时立即展示。

今天我们从 3 个角度展开:

  1. <link> 预加载(最简单,零 JS 方案)
  2. JS fetch 预加载(更灵活,可控)
  3. fetch + History API(支持前进/后退按钮,最佳体验)

HTML5 提供了原生预取机制,写在 <head> 里:

ini 复制代码
<link rel="prefetch" href="/chapter2.html" as="document">
  • prefetch:浏览器空闲时预加载资源(下一章 HTML / 图片等)。
  • prerender:甚至会在后台完整渲染页面,点击时几乎零延迟,但兼容性较差。

👉 优点:

  • 实现简单,一行代码搞定
  • 浏览器帮你做缓存

👉 缺点:

  • 无法控制何时加载(由浏览器决定)
  • 不会自动和前进/后退联动

适用场景:简单优化,比如普通博客或小说目录页。


三、方式二:JS fetch 主动预加载

如果我们想要更精确的控制,可以用 fetch

ini 复制代码
let nextUrl = "/chapter2.html";
let cache = null;

// 页面加载时主动预取下一章
fetch(nextUrl)
  .then(res => res.text())
  .then(html => cache = html);

document.getElementById("next").addEventListener("click", (e) => {
  e.preventDefault();
  if (cache) {
    document.getElementById("content").innerHTML = cache;
  } else {
    location.href = nextUrl;
  }
});

👉 优点:

  • 精确可控,随时预取
  • 可以缓存多章内容

👉 缺点:

  • 前进/后退按钮失效(因为没有写入历史栈)

四、方式三:fetch + History API(推荐🔥)

要让「前进/后退」也能秒开,就必须结合 pushState + popstate

ini 复制代码
const contentEl = document.getElementById("content");
let preloadCache = {};

// 初始化:写入当前页面内容
history.replaceState({ html: contentEl.innerHTML }, "", location.href);

// 预取下一章
function preload(url) {
  if (preloadCache[url]) return;
  fetch(url).then(res => res.text()).then(html => {
    const doc = new DOMParser().parseFromString(html, "text/html");
    const nextContent = doc.querySelector("#content").innerHTML;
    preloadCache[url] = nextContent;
    // 让历史记录也存上
    history.replaceState({ ...history.state, [url]: nextContent }, "", location.href);
  });
}

// 点击切换
document.querySelectorAll("a.nav").forEach(link => {
  link.addEventListener("click", async (e) => {
    e.preventDefault();
    const url = link.href;

    let newContent = preloadCache[url];
    if (!newContent) {
      const res = await fetch(url);
      const html = await res.text();
      const doc = new DOMParser().parseFromString(html, "text/html");
      newContent = doc.querySelector("#content").innerHTML;
    }

    contentEl.innerHTML = newContent;
    history.pushState({ html: newContent }, "", url);

    // 顺便预取下一章
    const next = link.nextElementSibling?.href;
    if (next) preload(next);
  });
});

// 前进/后退
window.addEventListener("popstate", (e) => {
  if (e.state && e.state.html) {
    contentEl.innerHTML = e.state.html;
  } else {
    location.reload();
  }
});

👉 优点:

  • 点击秒开
  • 前进/后退秒开
  • 可扩展预取策略

👉 缺点:

  • 需要额外 JS 逻辑
  • 刷新时依然需要服务端响应

五、对比总结

方案 代码复杂度 可控性 前进/后退支持 适用场景
<link rel="prefetch"> 极低 简单优化
JS fetch 定制预取策略
fetch + History API 小说网站 / 阅读器

六、进一步优化思路

  • 条件预取 :根据 navigator.connection.saveData 判断用户是否需要省流量。
  • 多章节预取:提前缓存后两章,点击前进时也能秒开。
  • Service Worker:离线缓存,断网也能继续阅读。
  • 无限滚动:直接拼接章节,去掉翻页操作。

七、总结

  • 最简单的方式<link rel="prefetch">,无需 JS。
  • 更灵活的方式 :JS fetch,可缓存下一章。
  • 最佳体验的方式fetch + History API,前进/后退全支持。

👉 对于小说网站,推荐 第三种方案,配合预加载和缓存,可以做到媲美本地阅读器的流畅体验。

相关推荐
周星星日记2 分钟前
vue中hash模式和history模式的区别
前端·面试
Light602 分钟前
Vue 高阶优化术:v-bind 与 v-on 的实战妙用与思维跃迁
前端·低代码·vue3·v-bind·组件封装·v-on·ai辅助开发
周星星日记3 分钟前
5.为什么vue中使用query可以保留参数
前端·vue.js
lebornjose3 分钟前
javascript - webgl中绑定(bind)缓冲区的逻辑是什么?
前端·webgl
瘦的可以下饭了8 分钟前
Day05- CSS 标准流、浮动、Flex布局
前端
前端无涯10 分钟前
React中setState后获取更新后值的完整解决方案
前端·react.js
西愚wo10 分钟前
前端开发者必备:在浏览器控制台批量提取HTML表单字段名(Label)
前端
小鸡吃米…37 分钟前
Python - 类属性
java·前端·python
前端不太难1 小时前
Navigation State 驱动的页面调试方法论
开发语言·前端·react.js
用户47949283569151 小时前
你每天都在用的 JSON.stringify ,V8 给它开了“加速通道”
前端·chrome·后端