从零开始使用 **Taki + Node.js** 实现动态网页转静态网站的完整代码方案

以下是从零开始使用 Taki + Node.js 实现动态网页转静态网站的完整代码方案,包含预渲染、自动化构建、静态托管及优化功能:

一、环境准备

1. 初始化项目
mkdir static-site && cd static-site
npm init -y
2. 安装依赖
npm install taki express fs-extra path

二、完整代码 (generate.js)

const { request, cleanup } = require('taki');
const express = require('express');
const fs = require('fs-extra');
const path = require('path');// 配置参数
const config = {dynamicSiteUrl: 'http://localhost:3000', // 动态网站本地运行地址outputDir: path.join(__dirname, 'dist'), // 静态文件输出目录routes: ['/', '/about', '/product/:id'], // 需静态化的路由(支持动态参数)puppeteerOptions: { headless: "new" }, // 使用新版无头模式resourceFilter: req => !['image', 'font'].includes(req.resourceType()), // 过滤非关键资源retries: 3 // 失败重试次数
};// 1. 生成静态页面核心逻辑
async function generateStaticPage(url, outputPath) {let retry = 0;while (retry < config.retries) {try {const html = await request({url,wait: 2000, // 等待页面渲染puppeteerOptions: config.puppeteerOptions,resourceFilter: config.resourceFilter,htmlSelector: 'body', // 仅抓取body内容(可选)manually: true // 手动触发快照(等待异步加载)});await fs.outputFile(outputPath, html);console.log(`✅ 生成成功: ${path.basename(outputPath)}`);return;} catch (err) {retry++;console.error(`❌ 失败重试 ${retry}/${config.retries}: ${err.message}`);}}throw new Error(`页面生成失败: ${url}`);
}// 2. 批量生成静态文件
async function generateAllPages() {await fs.emptyDir(config.outputDir); // 清空旧文件for (const route of config.routes) {const dynamicParam = route.match(/:\w+/g)?.[0] || '';const fileName = route.replace(/:\w+/g, '[param]') + '.html';const outputPath = path.join(config.outputDir, fileName);const fullUrl = `${config.dynamicSiteUrl}${route}`;await generateStaticPage(fullUrl, outputPath);}
}// 3. 启动静态服务器
function startServer() {const app = express();const port = 3001;// 托管静态资源(带缓存优化)app.use(express.static(config.outputDir, {maxAge: '1d',setHeaders: (res) => res.set('Cache-Control', 'public, max-age=86400')}));// 处理SPA路由重定向app.get('*', (req, res) => {res.sendFile(path.join(config.outputDir, 'index.html'));});app.listen(port, () => {console.log(`🚀 静态服务器运行于 http://localhost:${port}`);});
}// 4. 主流程控制
(async () => {try {await generateAllPages();startServer();} catch (err) {console.error('🔥 严重错误:', err);process.exit(1);} finally {await cleanup(); // 释放Puppeteer资源}
})();

三、使用说明

1. 运行动态网站

确保你的 React/Vue 等动态网站在本地 http://localhost:3000 运行。

2. 启动静态生成
node generate.js
3. 访问静态站点

打开浏览器访问 http://localhost:3001,所有页面将以静态形式呈现。


四、进阶功能扩展

1. 动态参数处理(示例)

若路由为 /product/:id,将生成 /product/[param].html,Express 会自动匹配如 /product/123 的请求。

2. SEO 优化

generateAllPages 函数末尾添加:

// 生成sitemap.xml
const sitemap = `<?xml version="1.0" encoding="UTF-8"?>
<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9">${config.routes.map(route => `<url><loc>https://your-domain.com${route.replace(/:\w+/g, '')}</loc></url>`).join('')}
</urlset>`;
await fs.writeFile(path.join(config.outputDir, 'sitemap.xml'), sitemap);
3. 自动化部署脚本 (deploy.sh)
#!/bin/bash
node generate.js
tar -czvf dist.tar.gz dist/
scp dist.tar.gz user@server:/var/www/html
ssh user@server "tar -xzvf /var/www/html/dist.tar.gz"

五、技术要点解析

  1. Taki 核心能力

    • 基于 Puppeteer 实现无头浏览器渲染,抓取动态内容139
    • 支持资源过滤(过滤图片/字体)提升生成速度139
  2. Express 优化

    • 静态资源托管 + 缓存控制,提升访问速度4770
    • SPA 路由重定向解决 History 模式 404 问题25
  3. 可靠性设计

    • 失败重试机制应对网络波动23
    • 自动清理旧文件避免冗余34

六、与其他方案对比

方案适用场景优势工具链推荐
Taki 预渲染SPA/动态内容快速静态化无需改源码,支持复杂交互Taki + Express
Next.js SSG新项目开发增量生成,开发体验好Next.js + Vercel
纯静态生成器内容驱动型站点(博客)生成速度快,适合 MarkdownHugo/Jekyll

七、常见问题

  1. 图片路径错误

    • 在 Taki 配置中添加资源替换逻辑:
      html = html.replace(/src="\/assets\//g, 'src="assets/');
      
  2. 动态内容更新

    • 结合 Webhook 触发定时重新生成

完整代码参考:Taki 官方文档

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

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

相关文章

商业智能BI分析中,汽车4S销售行业的返厂频次有什么分析价值?

买过车的朋友会发现&#xff0c;同一款车不管在哪个4S店去买&#xff0c;基本上价格都相差不大。即使有些差别&#xff0c;也是带着附加条件的&#xff0c;比如要做些加装需要额外再付一下费用。为什么汽车4S销售行业需要商业智能BI&#xff1f;就是因为在汽车4S销售行业&#…

静态链接过程发生了什么?

在静态链接过程中主要发生了两件事。一是空间与地址分配&#xff0c;链接器扫描所有输入文件的段&#xff0c;合并相似段并且重新计算段长度和在虚拟内存中的映射关系&#xff0c;收集所有的符号放到全局符号表中。二是符号解析与重定位&#xff0c;链接器收集所有的段信息和重…

✎ 一次有趣的经历

&#x1f4c6;2025年3月17日 | 周一 | ☀️晴 &#x1f4cd;今天路过学院楼7&#xff0c;见到了满园盛开的花&#x1f33a;&#xff0c;心情瞬间明朗&#xff01; &#x1f4cc;希望接下来的日子也能像这些花一样&#xff0c;充满活力&#x1f525;&#xff01; &#x1…

docker安装redis

第一步&#xff1a;docker拉取redis镜像 这种命令如果没有指定版本则是最新版本&#xff1a;docker pull redis 成功了 docker images 查询已经拉取成功镜像 然后因为在容器内部我们修改redis的配置不好修改&#xff0c;所以我们可以进行挂载配置文件 这个配置文件可以方便…

C语言学习笔记(第三部份)

说明&#xff1a;由于所有内容放在一个md文件中会非常卡顿&#xff0c;本文件将接续C_1.md文件的第三部分 整型存储和大小端 引例&#xff1a; int main(void) {// printf("%d\n", SnAdda(2, 5));// PrintDaffodilNum(10000);// PrintRhombus(3);int i 0;int arr[…

Cortical Labs公司CL1人脑芯片:开启生物智能计算新时代

Cortical Labs公司CL1人脑芯片&#xff1a;开启生物智能计算新时代 在科技飞速发展的今天&#xff0c;人工智能已经深入到我们生活的各个角落&#xff0c;但随着其发展&#xff0c;也面临着能耗高、效率有限等诸多挑战。为了突破这些瓶颈&#xff0c;科学家们开始探索将生物学…

Python学习第十八天

Django模型 定义&#xff1a;模型是 Django 中用于定义数据库结构的 Python 类。每个模型类对应数据库中的一张表&#xff0c;类的属性对应表的字段。 作用&#xff1a;通过模型&#xff0c;Django 可以将 Python 代码与数据库表结构关联起来&#xff0c;开发者无需直接编写 S…

Windows 图形显示驱动开发-WDDM 3.0功能- 硬件翻转队列(一)

WDDM 3.0 之前的翻转队列模型 许多新式显示控制器支持对按顺序显示的多个帧排队的能力。 从 WDDM 2.1 开始&#xff0c;OS 支持将在下一个 VSync 中显示的多个未完成的翻转覆盖请求。 显示微型端口驱动程序 (KMD) 通过 DXGK_DRIVERCAPS 中的 MaxQueuedMultiPlaneOverlayFlipVS…

《Python深度学习》第二讲:深度学习的数学基础

本讲来聊聊深度学习的数学基础。 深度学习听起来很厉害,其实它背后是一些很有趣的数学原理。本讲会用简单的方式解释这些原理,还会用一些具体的例子来帮助你理解。 2.1 初识神经网络 先从一个简单的任务开始:识别手写数字。 想象一下,你有一堆手写数字的图片,你想让计算…

车载DoIP测试 --- CANoe DoIP中如何配置路由激活请求中的 OEM 特定场(RoutingActivationWithOEMSpecific)

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 简单,单纯,喜欢独处,独来独往,不易合同频过着接地气的生活,除了生存温饱问题之外,没有什么过多的欲望,表面看起来很高冷,内心热情,如果你身…

JDBC数据库连接池技术详解——从传统连接方式到高效连接管理

1. 引言 在开发数据库应用时&#xff0c;我们通常需要与数据库建立连接并执行SQL语句。传统的JDBC连接方式虽然简单直接&#xff0c;但在高并发场景下容易带来性能问题&#xff0c;甚至导致系统崩溃。因此&#xff0c;引入数据库连接池&#xff08;Connection Pool&#xff09…

【工具类】PDF文件转图片

PDF文件转文件 1. 引入Maven依赖 主要使用了 pdfbox 包与 hutool 包。 pdfbox 负责 pdf 到图片的转换&#xff1b; hutool 负责文件读取转换。 <dependency><groupId>org.apache.pdfbox</groupId><artifactId>pdfbox</artifactId><version…

使用DeepSeek,优化斐波那契数函数,效果相当不错

下面这段代码定义了一个递归函数 fibonacci&#xff0c;用于计算第 n 个斐波那契数。 def fibonacci(n):if n < 1:return nelse:return fibonacci(n - 1) fibonacci(n - 2)虽然代码逻辑正确&#xff0c;但其性能较差&#xff0c;尤其是对于较大的 n 值&#xff0c;其复杂度…

Forward Looking Radar Imaging by Truncated Singular Value Decomposition 论文阅读

Forward Looking Radar Imaging by Truncated Singular Value Decomposition and Its Application for Adverse Weather Aircraft Landing 1. 论文的研究目标与意义1.1 研究目标1.2 实际问题与意义2. 论文的创新方法与公式解析2.1 信号建模与问题转化2.2 截断奇异值分解(TSVD)…

provide/inject源码实现

在 Vue 3 中&#xff0c;provide 和 inject 是通过 Vue 的响应式系统和组件实例机制实现的&#xff0c;底层是依赖 Vue 3 中的 Proxy 和 Reactive 来实现跨层级的数据传递和响应式绑定。以下是一个简化版的实现逻辑&#xff0c;帮助理解 Vue 3 中 provide 和 inject 是如何实现…

Unix时间戳BKP备份寄存器RTC实时时钟

Unix时间戳 Unix时间戳&#xff0c;也称为POSIX时间或Epoch时间&#xff0c;是一种在Unix和类Unix操作系统中使用的时间表示方法。它表示的是自1970年1月1日00:00:00 UTC&#xff08;协调世界时&#xff09;至当前时间经过的秒数&#xff0c;不考虑闰秒。Unix时间戳通常以秒为…

【Linux内核系列】:进程板块与文件板块的综合

&#x1f525; 本文专栏&#xff1a;Linux &#x1f338;作者主页&#xff1a;努力努力再努力wz &#x1f4aa; 今日博客励志语录&#xff1a; 人生中成功只是一时的&#xff0c;失败却是人生的主旋律&#xff0c;但是如何面对失败却把人分成了不同的样子&#xff0c;有的人会被…

CellOracle|基因扰动研究基因功能|基因调控网络+虚拟干预

在gzh“生信小鹏”同步文章 论文来源: 发表期刊:Nature发表时间:2023年2月23日论文题目:Dissecting cell identity via network inference and in silico gene perturbation研究团队:Kenji Kamimoto 等,华盛顿大学医学院1. 研究背景与问题提出 细胞身份(Cell Identit…

专线、云 和 物联网(IoT)

专线、云 和 物联网&#xff08;IoT&#xff09; 是现代信息与通信技术&#xff08;ICT&#xff09;领域的三大重要组成部分&#xff0c;它们在企业和个人的数字化转型中扮演着关键角色。以下是对这三者的详细介绍及其相互关系&#xff1a; 1. 专线&#xff08;Leased Line&…

[Lc14_priority_queue] 最后一块石头重量 | 数据流中的第 K 大元素 | 前K个高频单词 | 数据流的中位数

目录 1.最后一块石头的重量 题解 2.数据流中的第 K 大元素 题解 3.前K个高频单词 题解 代码 ⭕4.数据流的中位数 题解 在C中&#xff0c;使用标准库中的priority_queue&#xff0c;默认情况下它是一个最大堆&#xff08;即大堆排序&#xff09;&#xff0c;这意味着最…