阅读视图

发现新文章,点击刷新页面。

跨越边界的艺术:现代 Web 开发跨域解决方案终极指南

一、跨域的本质:同源策略是什么?

想要解决跨域问题,首先要明白“跨域”从何而来。

1. 同源策略的定义

浏览器的同源策略(Same-Origin Policy) 是跨域的核心根源,它是浏览器最核心也最基本的安全功能。
所谓“同源”,要求两个页面的以下三点必须完全相同:

  • 协议(http/https)
  • 域名(包括主域名、子域名)
  • 端口号(80/443/3000等)

image.png 只要三者有其一不同,就会被判定为“跨域”。此时,浏览器会限制非同源页面的以下行为:

  • 读取非同源网页的 Cookie、LocalStorage、IndexedDB 等存储数据。
  • 获取非同源网页的 DOM 元素。
  • 向非同源地址发送 AJAX 请求(XMLHttpRequest/fetch)。

2. 为什么需要同源策略?

同源策略就像一道“防火墙”,从根本上限制了恶意网站的非法操作,保障了用户的信息安全。试想一下,如果没有同源策略:

  • 恶意网站可以轻易读取你网银页面的 Cookie,盗取账户信息。
  • 钓鱼网站可以嵌入真实的电商页面,篡改支付金额。
  • 任意网站都能向你的服务器发送伪造请求,发起 CSRF 攻击。

3. 跨域的常见场景

日常开发中,跨域几乎无处不在:

  • 前后端分离项目:前端运行在 localhost:5173,后端接口在 localhost:3000(端口不同)。
  • 调用第三方接口:如支付、地图、天气等第三方服务(域名不同)。
  • 多端协作:公司内部不同部门的系统对接(子域名不同)。

二、方案 1:JSONP——兼容性拉满的“老古董”

JSONP(JSON with Padding)是跨域方案中的“老前辈”,也是早期前端解决跨域最常用的方式,最大的优势是浏览器兼容性极好(甚至能兼容 IE6/7)。

1. JSONP 的核心原理

浏览器的同源策略限制了 AJAX 请求,但并没有限制 <script> 标签的 src 属性。<script> 可以加载任意域名的资源(比如 CDN 上的 jQuery)。
JSONP 正是利用这一“漏洞”实现跨域:

  • 前端动态创建 <script> 标签,通过 src 向跨域接口发送请求,同时传递一个回调函数名。
  • 后端接收到请求后,将数据包裹在回调函数中返回(即“JSON with Padding”)。
  • 前端的回调函数被执行,从而拿到跨域数据。

2. JSONP 实战实现

前端代码(封装 JSONP 函数)
这段代码封装了一个返回 Promise 的 JSONP 函数,便于处理异步逻辑:

// 封装JSONP请求函数,返回Promise方便异步处理
function jsonp({ url, params, callback }) {
  return new Promise((resolve, reject) => {
    // 1. 创建script标签
    let script = document.createElement('script')
    // 2. 定义全局回调函数,接收后端返回的数据
    window[callback] = function(data) {
      resolve(data) // 成功拿到数据,resolve Promise
      document.body.removeChild(script) // 移除script标签,避免污染
    }
    // 3. 拼接请求参数(包含回调函数名)
    params = { ...params, callback } // 比如:{wd: 'test', callback: 'show'}
    let arrs = []
    for (let key in params) {
      arrs.push(`${key}=${params[key]}`)
    }
    // 4. 设置script的src属性,发送请求
    script.src = `${url}?${arrs.join('&')}`
    document.body.appendChild(script)
    // 5. 处理请求失败场景
    script.onerror = function() {
      reject(new Error('JSONP请求失败'))
      document.body.removeChild(script)
    }
  })
}

// 调用JSONP请求
jsonp({
  url: 'http://localhost:3000/say',
  params: { wd: 'Iloveyou' },
  callback: 'show'
}).then(data => {
  console.log('JSONP请求结果:', data)
}).catch(err => {
  console.error(err)
})

后端代码(Node.js 原生实现)
后端需要接收回调函数名,并将数据包裹在函数调用中返回:

const http = require('http');
const server = http.createServer((req, res) => {
  // 匹配/say接口
  if (req.url.startsWith('/say')) {
    // 解析URL参数
    const url = new URL(req.url, `http://${req.headers.host}`);
    const callback = url.searchParams.get('callback'); // 获取回调函数名
    
    // 设置响应头:返回JS脚本
    res.writeHead(200, { 'Content-type': 'text/javascript' });
    // 构造返回数据,包裹在回调函数中
    const data = {
      id: 1,
      username: 'admin',
      msg: 'JSONP请求成功'
    }
    // 核心:返回 "回调函数(数据)" 格式的JS代码
    res.end(`${callback}(${JSON.stringify(data)})`);
  } else {
    res.writeHead(404);
    res.end('Not Found')
  }
})

server.listen(3000, () => {
  console.log('JSONP服务器运行在 http://localhost:3000');
})

3. JSONP 的优缺点

表格

维度 详细说明
✅ 优点 兼容性极强:支持所有主流浏览器,包括低版本 IE。 实现简单:无需复杂的配置,前端后端少量代码即可完成。
❌ 缺点 仅支持 GET 请求:因为 <script> 标签的 src 只能发起 GET 请求。 安全风险:容易遭受 XSS 攻击(加载的脚本可能包含恶意代码),需确保请求的服务器是可信的。 性能问题:额外加载的 <script> 标签会阻塞页面渲染,影响首屏加载速度。

4. 适用场景

仅推荐在兼容老旧浏览器(如需要支持 IE6/7)的场景下使用。在现代项目中,优先选择其他方案。

三、方案 2:CORS——现代跨域的主流之选

CORS(跨域资源共享,Cross-Origin Resource Sharing)是W3C制定的标准,也是目前解决跨域问题最主流、最推荐的方式。它本质上是在HTTP协议之上,通过增加特定的请求头和响应头,让浏览器和服务器协同工作,来判断一个跨域请求是否被允许。

1. CORS 的核心原理

CORS 的核心思想是:将跨域的控制权从浏览器完全转移到服务器。

1. 浏览器发起请求:当浏览器发起一个跨域请求时,它会自动在请求头中添加 Origin 字段,标明请求的来源(协议、域名、端口)。

2. 服务器决策:服务器接收到请求后,根据 Origin 字段的值来判断是否允许这个来源的请求。

3. 服务器响应:如果服务器允许该请求,它会在响应头中添加 Access-Control-Allow-Origin 字段,其值就是被允许的源。

4. 浏览器检查:浏览器收到响应后,会检查响应头中的 Access-Control-Allow-Origin 是否与请求的 Origin 匹配。如果匹配,则将响应数据返回给前端JS代码;如果不匹配,则浏览器会拦截响应,并在控制台抛出跨域错误。

2. 简单请求 vs 预检请求

CORS 将跨域请求分为两类:简单请求预检请求

简单请求 一个请求要成为简单请求,必须同时满足以下条件:

请求方法是以下之一:GET, HEAD, POST

自定义请求头:除了浏览器自动设置的 Accept, Accept-Language, Content-Language, Content-Type 等,没有添加其他自定义请求头。

Content-Type 的值仅限于:application/x-www-form-urlencoded, multipart/form-data, text/plain

对于简单请求,浏览器会直接发送,并在请求头中带上 Origin

预检请求 不满足简单请求条件的,就是预检请求。例如,使用 PUTDELETE 方法,或者 Content-Typeapplication/json,又或者添加了自定义请求头(如 token)。

对于预检请求,浏览器会先自动发起一个 OPTIONS 方法的请求(这就是“预检”),询问服务器是否允许当前的跨域请求。只有在服务器明确回复“允许”后,浏览器才会真正发起后续的请求。

3. CORS 实战实现

后端代码(Node.js + Express) 使用 Express 框架时,可以借助 cors 中间件轻松实现 CORS。

const express = require('express');
const cors = require('cors'); // 引入cors中间件
const app = express();

// 1. 允许所有来源的跨域请求 (最宽松的配置)
app.use(cors());

// 2. 或者,进行更精细的配置
const corsOptions = {
  origin: 'http://localhost:5173', // 只允许这个源访问
  methods: ['GET', 'POST', 'PUT'], // 允许的请求方法
  allowedHeaders: ['Content-Type', 'Authorization'], // 允许的自定义请求头
  credentials: true // 允许携带Cookie等凭证
};
app.use(cors(corsOptions));

// 定义一个需要跨域访问的接口
app.get('/api/data', (req, res) => {
  res.json({ message: 'CORS请求成功!', data: [1, 2, 3] });
});

app.listen(3000, () => {
  console.log('CORS服务器运行在 http://localhost:3000');
});

后端代码(Node.js 原生实现) 如果不使用框架,也可以手动设置响应头。

const http = require('http');
const server = http.createServer((req, res) => {
  // 设置允许跨域的源,* 表示允许所有源
  res.setHeader('Access-Control-Allow-Origin', '*');
  // 允许的请求方法
  res.setHeader('Access-Control-Allow-Methods', 'GET, POST, PUT, DELETE');
  // 允许的自定义请求头
  res.setHeader('Access-Control-Allow-Headers', 'Content-Type, Authorization');
  // 允许携带凭证(如Cookie)
  res.setHeader('Access-Control-Allow-Credentials', 'true');

  // 处理预检请求
  if (req.method === 'OPTIONS') {
    res.writeHead(204); // 204 No Content
    res.end();
    return;
  }

  // 处理其他业务请求
  if (req.url === '/api/data') {
    res.writeHead(200, { 'Content-Type': 'application/json' });
    res.end(JSON.stringify({ message: 'CORS请求成功!', data: [1, 2, 3] }));
  }
});

server.listen(3000, () => {
  console.log('CORS服务器运行在 http://localhost:3000');
});

4. 关键的 CORS 响应头

响应头 说明
Access-Control-Allow-Origin 必需。指定允许访问的源,可以是 *(通配符)或具体的源(如 http://example.com)。
Access-Control-Allow-Methods 预检请求必需。指定允许的请求方法,如 GET, POST, PUT
Access-Control-Allow-Headers 预检请求必需。指定允许的自定义请求头,如 Content-Type, Authorization
Access-Control-Allow-Credentials 可选。一个布尔值,表示是否允许浏览器发送 Cookie。如果为 true,则 Access-Control-Allow-Origin 不能为 *,必须是具体的源。
Access-Control-Max-Age 可选。指定预检请求的缓存时间(秒),避免频繁发送 OPTIONS 请求。

5. CORS 的优缺点

维度 详细说明
优点 功能强大:支持所有类型的 HTTP 请求(GET, POST, PUT, DELETE 等)。安全性高:通过服务器精确控制允许的源、方法和请求头。现代标准:被所有现代浏览器支持,是前后端分离项目的最佳实践。
缺点 需要后端配合:必须在服务器端进行配置,前端无法单方面解决。配置复杂:对于需要携带凭证或复杂请求头的场景,配置相对繁琐。兼容性问题:不支持 IE9 及以下版本。

6. 适用场景

CORS 是现代 Web 开发中解决跨域问题的首选方案,尤其适用于前后端分离的架构。只要后端能够配合修改响应头,就应该优先使用 CORS。

四、方案 3:反向代理——“曲线救国”的万能钥匙

反向代理是开发环境中解决跨域问题最常用、也最省心的方法之一。它的核心思想是 “曲线救国” :既然浏览器禁止前端直接访问后端接口,那我们就让前端请求一个和自己“同源”的代理服务器,再由这个代理服务器去请求真正的后端接口。

1. 反向代理的核心原理

1.  前端请求代理:前端应用(如运行在 localhost:5173)不再直接请求后端接口(如 localhost:3000/api),而是请求一个与自己同源的代理地址(如 localhost:5173/api)。因为源相同,所以不会触发浏览器的跨域限制。

2.  代理服务器转发:这个代理服务器(通常是开发服务器或 Nginx)收到请求后,会以自己的身份向真正的后端接口(localhost:3000/api)发起请求。这个请求是服务器与服务器之间的通信,不受浏览器同源策略的限制。

3. 代理服务器返回:代理服务器拿到后端接口的响应数据后,再原封不动地返回给前端。 通过这种方式,前端巧妙地绕过了浏览器的跨域限制,实现了数据的获取。

2. 反向代理实战实现

反向代理的实现方式多种多样,从开发环境的配置到生产环境的部署,都有其身影。

开发环境:Vite/Webpack 配置 在现代前端框架(如 Vue, React)的开发环境中,我们通常使用 Vite 或 Webpack 作为开发服务器。它们都内置了强大的代理功能,只需几行配置即可解决跨域问题。

Vite 配置示例 ( vite.config.js )

import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'

export default defineConfig({
  plugins: [vue()],
  server: {
    // 配置代理规则
    proxy: {
      // 当前端请求 /api 路径时,触发代理
      '/api': {
        target: 'http://localhost:3000', // 代理的目标服务器地址
        changeOrigin: true, // 修改请求头中的 Origin 为目标服务器的 Origin
        // rewrite: (path) => path.replace(/^/api/, '') // 可选:重写路径,去掉 /api 前缀
      }
    }
  }
})

配置完成后,前端请求 http://localhost:5173/api/data,开发服务器会自动将其转发到 http://localhost:3000/api/data

生产环境:Nginx 配置 在项目部署到生产环境时,Nginx 是最常用的反向代理服务器。它不仅能处理跨域,还能提供负载均衡、静态资源服务、缓存等多种功能。

Nginx 配置示例 ( nginx.conf )

server {
    listen       80;
    server_name  localhost; # 或者你的域名

    # 1. 配置前端静态文件
    location / {
        root   /usr/share/nginx/html; # 前端打包文件的路径
        index  index.html index.htm;
        try_files $uri $uri/ /index.html; # 解决前端路由 history 模式刷新404的问题
    }

    # 2. 配置后端接口代理
    location /api/ {
        proxy_pass http://backend_server:3000/; # 后端服务器的地址
        proxy_set_header Host $host;             # 传递原始主机头
        proxy_set_header X-Real-IP $remote_addr; # 传递用户真实IP
        proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    }
}

这样配置后,无论是前端页面还是 /api/ 开头的接口请求,都由同一个 Nginx 服务器处理,完美规避了跨域问题。

3. 反向代理的优缺点

维度 详细说明
优点 前端无感:前端代码无需任何修改,完全不用关心跨域问题。功能强大:除了跨域,还能实现负载均衡、请求/响应拦截、日志记录等。通用性强:适用于任何类型的请求,不受请求方法和请求头的限制。
缺点 增加服务器成本:需要额外部署和维护一台代理服务器(如 Nginx)。配置相对复杂:相比 CORS,反向代理的配置(尤其是 Nginx)需要一定的运维知识。可能增加延迟:请求多了一次转发,理论上会增加一点点网络延迟。

4. 适用场景

● 开发环境:强烈推荐使用 Vite/Webpack 的代理功能,是开发阶段解决跨域问题的首选。

● 生产环境:当你无法控制后端服务器(例如调用第三方API),或者后端团队不方便配合配置 CORS 时,使用 Nginx 反向代理是最佳选择。它也是微服务架构中 API 网关的雏形。

五、方案 4:WebSocket——实时通信的“特权通道”

WebSocket 是一种在单个 TCP 连接上进行全双工通信的协议。它最大的特点是不受同源策略的限制,这使得它在需要实时双向通信的场景下,成为了一个天然的跨域解决方案。

1. WebSocket 的核心原理

WebSocket 的工作流程可以分为三个阶段:

1. 握手阶段:前端通过 JavaScript 创建一个 WebSocket 对象,浏览器会向服务器发起一个特殊的 HTTP 请求。这个请求头中包含 Upgrade: websocket 字段,表示希望将协议从 HTTP 升级到 WebSocket。

2. 协议升级:服务器收到请求后,如果支持 WebSocket,会返回一个状态码为 101 (Switching Protocols) 的响应,同意协议升级。

3. 数据传输:一旦握手成功,客户端和服务器之间就建立了一条持久的 TCP 连接。此后,双方可以随时主动向对方推送数据,而无需像 HTTP 那样由客户端反复发起请求。

正是因为 WebSocket 在握手成功后就脱离了 HTTP 协议的范畴,建立了一条独立的“管道”,所以浏览器不会对其应用同源策略的限制。

2. WebSocket 实战实现

前端代码 前端使用非常简单,只需几行代码即可建立连接并监听事件。

// 1. 创建 WebSocket 连接,传入服务器地址
// 注意:协议是 ws:// (或 wss:// 用于加密连接)
const socket = new WebSocket('ws://localhost:3000');

// 2. 监听连接成功事件
socket.onopen = function(event) {
  console.log('WebSocket 连接已建立');
  // 连接成功后,可以立即向服务器发送数据
  socket.send(JSON.stringify({ type: 'init', data: 'Hello Server!' }));
};

// 3. 监听服务器发来的消息
socket.onmessage = function(event) {
  console.log('收到服务器消息:', event.data);
  const data = JSON.parse(event.data);
  // 根据消息类型处理数据
  if (data.type === 'notification') {
    alert(data.message);
  }
};

// 4. 监听连接关闭事件
socket.onclose = function(event) {
  console.log('WebSocket 连接已关闭');
};

// 5. 监听连接错误事件
socket.onerror = function(event) {
  console.error('WebSocket 发生错误:', event);
};

// 随时可以向服务器发送数据
function sendMsg() {
  socket.send('这是一条新消息');
}

后端代码(Node.js + ws 库) 后端可以使用 ws 这个轻量级的 WebSocket 库来快速搭建服务器。

const WebSocket = require('ws');

// 创建一个 WebSocket 服务器,监听 3000 端口
const wss = new WebSocket.Server({ port: 3000 });

// 监听客户端连接事件
wss.on('connection', (ws) => {
  console.log('有客户端连接进来了');

  // 向当前连接的客户端发送欢迎消息
  ws.send(JSON.stringify({ type: 'welcome', message: '欢迎连接到 WebSocket 服务器' }));

  // 监听当前客户端发来的消息
  ws.on('message', (data) => {
    console.log('收到客户端消息:', data.toString());
    // 可以将消息广播给所有连接的客户端
    wss.clients.forEach((client) => {
      if (client.readyState === WebSocket.OPEN) {
        client.send(`广播: ${data}`);
      }
    });
  });

  // 监听客户端断开连接
  ws.on('close', () => {
    console.log('客户端连接已关闭');
  });
});

console.log('WebSocket 服务器运行在 ws://localhost:3000');

3. WebSocket 的优缺点

维度 详细说明
优点 天然跨域:不受同源策略限制,无需额外配置。实时双向通信:服务器可以主动向客户端推送数据,延迟极低。持久连接:只需一次握手,即可保持长时间通信,减少了 HTTP 反复建立连接的开销。
缺点 协议不同:需要服务器和客户端都支持 WebSocket 协议,不适用于传统的 HTTP 请求场景。兼容性问题:不支持 IE9 及以下版本。连接管理复杂:需要处理连接的建立、维持、断开和重连,比简单的 HTTP 请求更复杂。

4. 适用场景

WebSocket 专为实时性要求高的场景而生,例如:

● 在线聊天/即时通讯:如微信网页版、在线客服。

● 实时数据推送:如股票行情、体育比赛比分、新闻快讯。

● 协同编辑:如在线文档、代码编辑器。

● 多人在线游戏:需要实时同步玩家状态。


六、方案 5:Vite 反向代理 —— 本地开发的 “最优解”

在前端本地开发阶段,我们经常会遇到这样的场景:前端项目运行在 http://localhost:5173,而后端接口运行在 http://localhost:3000。虽然这只是开发环境下的端口不同,但在浏览器看来这就是“跨域”。

虽然可以通过后端配置 CORS 来解决,但在开发阶段,更优雅、更安全的方式是利用开发服务器(如 Vite、Webpack)进行反向代理。这种方式不需要后端做任何改动,完全由前端工具链来处理跨域问题。

1. Vite 代理的核心原理

Vite 代理的本质是利用了开发服务器(Dev Server)作为“中间人”。

  • 前端视角:前端代码发起请求时,目标地址是 Vite 服务器(例如 /api/user)。
  • 同源策略豁免:因为 Vite 服务器就是提供前端页面的服务端,所以前端请求 /api/user 属于“同源请求”,浏览器不会拦截。
  • 服务器转发:Vite 服务器接收到请求后,发现这是一个代理请求,于是它会以“服务器身份”向真正的后端接口(例如 http://localhost:3000/api/user)发起请求。
  • 响应返回:后端接口将数据返回给 Vite 服务器,Vite 再将数据返回给前端浏览器。

关键点:浏览器与 Vite 服务器之间是同源的(无跨域);Vite 服务器与后端服务器之间是服务器间的通信(不受浏览器同源策略限制)。通过这种“曲线救国”的方式,完美绕过了浏览器的跨域限制。

2. Vite 代理实战配置

Vite 内置了强大的代理功能,基于 http-proxy 中间件实现。我们只需要在 vite.config.js 中进行简单的配置即可。

配置步骤:

  1. 打开项目根目录下的 vite.config.js 文件。
  2. 在 server 选项中添加 proxy 配置。
  3. 定义需要代理的路径前缀(如 /api)。
// vite.config.js
import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'

export default defineConfig({
  plugins: [vue()],
  server: {
    // 配置代理规则
    proxy: {
      // 1. 定义代理前缀
      // 当请求路径以 '/api' 开头时,触发代理
      '/api': {
        // 2. 目标服务器地址
        // 这里填写后端接口的真实地址
        target: 'http://localhost:3000',
        
        // 3. 是否改变请求头中的 Origin
        // 设置为 true 时,Vite 会将请求头的 Host 改为目标服务器的 Host
        // 避免后端因为 Origin 校验不通过而拒绝请求
        changeOrigin: true,
        
        // 4. 路径重写 (可选)
        // 如果后端不需要 '/api' 这个前缀,可以将其重写为空
        // 例如:前端请求 '/api/user' -> 后端接收 '/user'
        rewrite: (path) => path.replace(/^/api/, '')
      },
      
      // 5. 多个代理配置 (可选)
      // 如果有多个不同的后端服务,可以继续添加
      '/upload': {
        target: 'http://upload-server.com',
        changeOrigin: true
      }
    }
  }
})

3. 配置项详解

表格

配置项 类型 说明
target String 必需。你要代理到的目标地址,即后端接口的真实域名或 IP。
changeOrigin Boolean 推荐开启。设为 true 时,会自动修改请求头中的 host 为 target 的值。很多后端框架(如 Nginx、Java Spring)会校验 host,不开启可能导致 403/404 错误。
rewrite Function 可选。用于重写请求路径。例如,如果后端接口不需要前端定义的前缀(如 /api),可以用此函数将其替换或删除。
secure Boolean 如果目标是 https 接口,设为 false 可以忽略 HTTPS 证书校验(开发环境常用)。

4. 优缺点分析

优点:

  • 开发环境专用神器:无需后端配合,前端开发者自己就能搞定,不影响生产环境配置。
  • 无跨域风险:完全在开发服务器层面处理,浏览器根本感知不到跨域的存在。
  • 配置极其简单:Vite 内置功能,几行代码即可完成,且支持 TypeScript 配置。
  • 支持 WebSocket:Vite 代理也支持代理 WebSocket 连接(配置 ws: true)。

缺点:

  • 仅限开发环境:Vite 代理只在 vite dev 启动的开发服务器中生效。项目打包上线后,Vite 服务器不再运行,代理配置也随之失效。
  • 无法解决生产环境问题:它只是一个开发时的“模拟器”,不能用于解决线上环境的跨域问题。

5. 适用场景

  • 前端本地开发:这是该方案的唯一且最佳场景。
  • 接口联调阶段:在后端尚未部署或无法修改响应头时,前端通过代理快速进行联调。
  • Mock 数据切换:配合环境变量,可以在代理真实接口和本地 Mock 服务之间灵活切换。

八、方案 7:postMessage —— 跨域通信的“万能信使”

前文提到的 JSONP、CORS、反向代理等方案,主要解决的是“浏览器向服务器请求数据”的跨域问题。但在现代 Web 开发中,我们经常会遇到“页面与页面”、“窗口与窗口”之间需要通信的场景,例如:

  • 父页面与嵌入的跨域 iframe 进行数据交互。
  • 主窗口与通过 window.open() 打开的跨域子窗口同步状态。
  • 主线程与 Web Worker 之间传递消息。

对于这些场景,postMessage 是 HTML5 提供的标准解决方案,它就像一个“万能信使”,允许不同源的窗口之间安全地进行双向通信。

1. postMessage 的核心原理

postMessage 的核心思想是 “消息传递” 而非“直接访问”。它打破了浏览器的同源策略,但并没有完全移除安全限制。

  • 发送方:通过 targetWindow.postMessage(message, targetOrigin) 方法,向目标窗口发送一条结构化的数据消息。
  • 接收方:通过监听 window 对象的 message 事件,来捕获并处理来自其他窗口的消息。
  • 安全校验:整个过程通过 targetOrigin(发送时指定目标源)和 event.origin(接收时检查消息来源)进行双重安全校验,确保消息只在你信任的窗口之间传递。

2. postMessage 实战实现

我们以最常见的 父页面与跨域 iframe 通信 为例,展示如何实现双向通信。

父页面 (parent.html)

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <title>父页面</title>
</head>
<body>
    <h1>我是父页面</h1>
    <!-- 嵌入一个跨域的 iframe -->
    <iframe id="childFrame" src="https://iframe-example.com/child.html"></iframe>

    <script>
        const iframe = document.getElementById('childFrame');

        // 1. 向 iframe 发送消息
        // 注意:必须等待 iframe 加载完成后再发送
        iframe.onload = () => {
            const data = { type: 'GREETING', text: 'Hello from parent!' };
            // 精确指定目标源,这是安全的关键!
            iframe.contentWindow.postMessage(data, 'https://iframe-example.com');
        };

        // 2. 监听来自 iframe 的消息
        window.addEventListener('message', (event) => {
            // 【安全红线】必须校验消息来源!
            if (event.origin !== 'https://iframe-example.com') {
                console.warn('收到来自非法源的消息,已忽略');
                return;
            }
            console.log('父页面收到消息:', event.data);
        });
    </script>
</body>
</html>

子页面 (child.html)

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <title>子页面 (iframe)</title>
</head>
<body>
    <h1>我是子页面 (iframe)</h1>
    <script>
        // 1. 监听来自父页面的消息
        window.addEventListener('message', (event) => {
            // 【安全红线】同样必须校验消息来源!
            if (event.origin !== 'https://parent-example.com') {
                return;
            }
            console.log('子页面收到消息:', event.data);

            // 2. 向父页面回复消息
            const replyData = { type: 'REPLY', text: 'Hello back!' };
            // event.source 是发送消息的窗口对象的引用
            event.source.postMessage(replyData, event.origin);
        });
    </script>
</body>
</html>

3. API 详解

发送消息:targetWindow.postMessage(message, targetOrigin)

表格

参数 类型 说明
message 任意类型 要发送的数据。可以是字符串、对象、数组等,数据会被浏览器使用“结构化克隆算法”进行序列化。
targetOrigin String 安全关键!  指定接收消息的窗口的源(协议+域名+端口)。必须精确指定,严禁在生产环境使用通配符 '*' ,否则可能导致敏感数据泄露给恶意网站。

接收消息:window.addEventListener('message', callback)

回调函数接收一个 MessageEvent 对象,其中包含三个关键属性:

表格

属性 类型 说明
event.data 任意类型 发送方传递的实际消息数据。
event.origin String 安全关键!  发送消息的窗口的源。接收方必须校验此属性,确保消息来自可信源。
event.source Window 对象 发送消息的窗口对象的引用。可用于向发送方回传消息,实现双向通信。

4. 优缺点分析

优点:

  • 功能强大:解决了页面间通信的跨域问题,这是 CORS 和代理无法做到的。
  • 双向通信:支持父子窗口、主副窗口之间的双向消息传递。
  • 安全性高:通过 targetOrigin 和 event.origin 的双重校验,可以有效防止恶意攻击。
  • 数据灵活:支持传递复杂的结构化数据。

缺点:

  • 使用场景特定:仅适用于窗口间通信,不适用于常规的 AJAX 请求。
  • 安全要求高:开发者必须手动进行源校验,任何疏忽都可能导致严重的安全漏洞(如 XSS)。
  • 异步通信:基于事件模型,处理复杂交互时逻辑可能变得分散。

5. 适用场景

  • 第三方组件集成:如嵌入支付宝/微信支付的 iframe,支付完成后通知父页面。
  • 跨域单点登录(SSO) :通过一个中央登录页,使用 postMessage 将登录令牌传递给其他域名的应用。
  • 微前端架构:主应用与子应用之间进行状态同步和事件通知。
  • 多窗口协作:如在线协作文档,主窗口打开多个编辑窗口,并同步光标位置和编辑内容。
  • Web Worker:主线程与后台线程进行数据交换。

HTTP 演进史:每次升级都在解决什么痛点?

模块一:引言

HTTP(HyperText Transfer Protocol) 是基于 TCP/IP 的应用层协议,是互联网数据通信和网页传输的基石。

它采用请求-响应模式,是无状态协议,特别适合海量用户的高 并发访问场景。

HTTP 是一种请求-响应模式的协议:客户端发起请求,服务器返回响应。过程非常 简单,却极其强大——正是这种简洁性,让 HTTP 得以快速普及,成为互联网的"通用语言"。

同时,它也是无状态协议——服务器不会记得你上一次请求说了什么。每次请求都是 独立的。这种设计让它能够轻松应对海量并发访问,但也催生了 Session、Cookie、Token 等会话管理机制。

从 1991 年 HTTP 0.9 诞生至今,经历了多次重大升级。本文 将从历史演进的角度,带你系统回顾 HTTP 协议的发展脉络。

模块二:HTTP 0.9 和 HTTP 1.0 — 协议的诞生与初探

HTTP 0.9(1991年)— 一切的开始

1991年,万维网(World Wide Web)的发明者 Tim Berners-Lee 发布了 HTTP
的第一个版本。

这是一个极其简单的协议,只有一行命令:

GET /index.html

服务器直接返回 HTML 文档内容,传输完成就断开连接。

它的特点:

  • 仅支持 GET 方法
  • 无请求头、无响应头
  • 只能传输纯 HTML 文本
  • 每次请求都是一个新的 TCP 连接

现在的眼光看,HTTP 0.9 简陋得难以置信。但正是这个简单的开始,开启了互联网时代的大门。

HTTP 1.0(1996年)— 走向标准化

随着互联网蓬勃发展,HTTP 0.9 已经不够用了。1996年,HTTP 1.0 正式发布,协议开始走向标准化。

1. 多种请求方法

HTTP 1.0 扩展了请求方法:

  • GET — 从服务器读取资源
  • POST — 向服务器提交数据
  • HEAD — 仅获取响应头,不返回正文

POST 的出现意义重大——它让表单提交成为可能,Web 开始从"只读"走向"可写"。

2. 请求头(Headers)

HTTP 1.0 创新性地引入了请求头和响应头的概念,让客户端和服务器能够传递元数据。

常用请求头:

  • User-Agent — 客户端信息,比如浏览器版本、操作系统
  • Cookie — 会话标识,服务端下发的会话 ID
  • Content-Type — 请求体格式,比如 application/json
  • Accept — 可接受的响应类型,比如 text/html, */
  • Authorization — 认证信息,比如 Bearer token 或 Basic auth

User-Agent 是其中最有意思的一个。它的格式大致如下:

Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/146.0.0.0 Safari/537.36

这一串标识的意思是:

image.png

  • Mozilla/5.0 最初是 Netscape 浏览器的标识,后来所有浏览器都兼容这个标识
  • (Macintosh; Intel Mac OS X 10_15_7) 告诉你操作系统和硬件信息
  • AppleWebKit/537.36 是渲染引擎
  • Chrome/146.0.0.0 是浏览器版本号
  • Safari/537.36 是为了兼容 Safari

这就是为什么国内早期要区分 PC 站和移动站——不同的 User-Agent 告诉服务器你用的是什么设备,服务器就返回不同的页面。

3. 短连接

HTTP 1.0 依然是短连接模式:每次请求都要先 TCP三次握手建立连接,请求完成后四次挥手断开连接,下一次请求再重新建连。

一个网页可能有几十个资源——HTML、CSS、JS、图片、字体……每个都要单独建立和断开 TCP 连接。这种方式在当时互联网规模还不大的时候勉强够用,但随着网页资源越来越多,性能问题开始凸显。

HTTP 1.0 为协议奠定了基本框架:确立了请求-响应模式,引入了 Headers
的概念,支持多种请求方法。 但它也有很多局限:短连接效率低、无状态导致会话管理困难、明文传输不安全…… 这些问题的解决方案,都留给了下一代的 HTTP 1.1。

模块四:HTTP 2.0 — 性能飞跃

HTTP 1.1 虽然大大提升了 Web的能力,但它的核心问题——对头阻塞——始终没有解决。

2015年,HTTP 2.0发布,专门针对这个痛点进行了底层重构。

1. 二进制分帧

HTTP 1.1 的数据是明文传输的,所有数据混在一起,没有边界,没有编号,想插队 是不可能的。

举个例子:

假设浏览器同时请求:

  - 流 1:index.html
  - 流 3:style.css
  - 流 5:app.js

  HTTP 2.0 的传输可能是这样的——帧交错在一起:

  流1帧头 → 流3帧头 → 流1数据 → 流5帧头 → 流3数据 → 流1数据 → ...

  到达接收端之后,按流 ID 分开重组:

  流1:帧1-1 + 帧1-2 + 帧1-3 → 拼成 index.html
  流3:帧3-1 + 帧3-2          → 拼成 style.css
  流5:帧5-1 + 帧5-2          → 拼成 app.js

为什么能解决对头阻塞?

因为每个流都有自己的 ID,独立重组。假设流 1 的某个帧丢了,只影响流1,流3和流5 照样传输、照样重组,完全不受影响。

2. 多路复用

基于二进制分帧,HTTP 2.0 实现了真正的多路复用(Multiplexing):

  • 一个 TCP 连接中可以并发多个请求
  • 每个请求都有一个独立的流 ID
  • 帧可以交错发送,按流 ID 归类重组

举个例子:

 浏览器要请求 index.html、style.css、app.js 三个资源。

  HTTP 1.1 时代:

  同一个域名只能开 1-6 个 TCP
  连接(浏览器限制),每个请求必须等上一个响应回来才能发下一个:

  TCP连接1:GET /index.html → 等 → 收到响应 → GET /style.css → 等 →        
  收到响应 → GET /app.js ...
  TCP连接2:GET /app.js     → 等 → 收到响应
  ...

  如果 index.html 卡住了,后面 style.css 和 app.js 只能在后面排队等。      

  HTTP 2.0 时代:

  一个 TCP 连接就够了。三个请求分属三个流,同时发送:

  一个 TCP 连接里:
  → 流1: GET /index.html
  → 流3: GET /style.css
  → 流5: GET /app.js
  ← 流1: 返回 index.html 数据帧
  ← 流3: 返回 style.css 数据帧
  ← 流5: 返回 app.js 数据帧

  三个流并行跑,互不等待,互不阻塞。

这从根本上解决了 HTTP 1.1 的对头阻塞问题

3. 服务器推送

传统的请求模式是:浏览器请求 HTML → 服务器返回 HTML → 浏览器解析 HTML
发现需要 CSS/JS → 再去请求 CSS/JS。

HTTP 2.0 支持服务器推送(Server Push):服务器知道浏览器需要什么资源,主动把 CSS/JS 推送给浏览器,浏览器还没请求就已经收到了。

这样就省去了浏览器反复请求的延迟。

4. 头部压缩

HTTP 2.0 还对 Header 进行了压缩。因为一个请求的 Header 往往有几百字节,而实际数据可能只有几字节,Header 的开销非常大。

HTTP 2.0 使用 HPACK 算法压缩 Header,进一步减少了传输量。


HTTP 2.0 的意义

HTTP 2.0 通过二进制分帧和多路复用,从根本上解决了 HTTP 1.1 的对头阻塞问题,让 Web 性能有了质的飞跃。

但它仍然有一个隐患——底层还是 TCP。TCP 在传输层也有对头阻塞问题,一旦丢包,整个 TCP 连接上的所有流都会受影响。

这个问题的最终解决方案,就是 HTTP 3.0。

模块五:HTTP 3.0 — QUIC 革命

HTTP 2.0 虽然解决了应用层的对头阻塞,但它的底层还是 TCP。TCP在传输层同样存在对头阻塞——一旦丢包,整个 TCP连接上的所有数据都要等待重传,所有流都被卡住。

HTTP 3.0 就是为了彻底解决这个问题。

HTTP 3.0 的核心是 QUIC(Quick UDP Internet Connections),一种基于 UDP的传输协议。

UDP 的特点是什么?无连接、不重传、不管顺序——简单粗暴,速度快,但不可靠。

QUIC

在 UDP 之上实现了自己的可靠传输逻辑,把 TCP 的优点移植过来,同时避免了
TCP 的缺点。

HTTP 3.0 的核心改进

1. 彻底抛弃 TCP,全程 UDP

HTTP 3.0 不再使用 TCP,而是直接基于 QUIC。没有 TCP 三次握手,改用 QUIC
自己的连接建立逻辑。

2. 每个流独立,不再互相影响

QUIC 把连接分成多个流(Stream)。丢包了?只影响当前流,其他流照常跑。这就 是真正的无对头阻塞。

3. 0-RTT 快速建立连接

第一次连接需要 1-RTT,之后可以做到 0-RTT——客户端直接发送数据,连接已经建立了。

4. 内置 TLS

HTTP 3.0 把加密直接做进了传输层,而不是像 HTTPS 那样单独跑一个 TLS
层。QUIC 本身就支持加密,而且比 TLS 更快。

一句话总结

HTTP 3.0 = HTTP 2.0 的所有特性 + QUIC(基于 UDP)= 彻底解决对头阻塞 +
更快建立连接 + 内置加密


模块六:总结

一部解决痛点的历史回顾 HTTP 的演进,有一条清晰的脉络:

 HTTP 0.9HTTP 1.0HTTP 1.1HTTP 2.0HTTP 3.0

每一次升级,都是为了解决上一代暴露出来的核心问题。


  • HTTP 0.9 太简陋,只能发 GET 请求,于是 HTTP 1.0 加入了 POST、HEAD、请求头和响应头。

  • HTTP 1.0 每次请求都要重新建连,效率太低,于是 HTTP 1.1 引入了长连接,多个请求可以复用同一个 TCP 连接。

  • HTTP 1.1 的管道化听起来很美,但响应没有编号,一个请求卡住后面全部排队,实际被浏览器弃用。

  • HTTP 2.0 用二进制分帧和流 ID 彻底解决了这个问题,一个 TCP 连接里可以并发多个请求,互不阻塞。

HTTP 2.0 看起来很完美了,但底层还是 TCP。TCP本身的传输层对头阻塞没有解决,一旦丢包,所有流都受影响。

  • HTTP 3.0 直接抛弃TCP,改用基于 UDP 的 QUIC,每个流独立可靠传输,彻底告别了对头阻塞。

各版本一句话定位

  • HTTP 0.9:一行 GET,一切的开端
  • HTTP 1.0:引入 Header,走向标准化
  • HTTP 1.1:长连接为主,但应用层对头阻塞无法根治
  • HTTP 2.0:二进制分帧 + 多路复用,从根本上解决对头阻塞
  • HTTP 3.0:基于 QUIC(UDP),彻底解决对头阻塞,更快、更安全

面试该怎么答

当面试官问起 HTTP 的演进时,不要只背区别,要讲清为什么需要这些升级。

比如被问到"HTTP 2.0 相比 1.1 有什么改进",可以这样答:

HTTP 1.1 虽然有长连接,但响应没有编号,存在对头阻塞问题。HTTP 2.0
通过二进制分帧和流 ID,把每个请求拆成带编号的帧,在同一个 TCP 连接里并发传输,按流 ID 重组,彻底解决了对头阻塞。

顺着"问题→解决方案→新问题→再解决"的逻辑讲下去

红绿灯也内卷?用 JS 给马路 “指挥家” 写个打工脚本

前言

每天见的红绿灯,红灯 3 秒、绿灯 2 秒、黄灯 1 秒的循环节奏,本质就是 JS 里经典的异步流程控制问题,也就是面试中的🚥红绿灯算法。今天咱们用极简的方式,给这个 “马路指挥家” 写段代码,让它规规矩矩按点 “上岗”。

一、先给红绿灯定个 “打工规则”

咱们先明确需求:红绿灯要按红→绿→黄的顺序循环亮灯,红灯工作 3 秒,绿灯 2 秒,黄灯 1 秒。核心难点在于 —— 必须等前一个灯亮完,下一个灯才能上岗,不能出现 “红绿同框” 的尴尬场面。

在 JS 里,要实现这种 “等一等再干活” 的逻辑,Promise是最佳拍档。先封装一个设置灯色和时长的函数,让每个灯的亮起都变成一个 “可等待” 的任务:

// 定义设置灯色的函数,返回Promise对象
function setColor(color, time) {
    return new Promise((resolve, reject) => {
        // 先打印当前灯的状态,告诉我们它要“上岗”多久
        console.log(`${color}灯亮起,需要等${time/1000}秒`);
        
        // 定时器模拟灯亮的时长,时间到了就“完成”这个任务
        setTimeout(() => {
            resolve(color); // 任务完成,返回当前灯色
        }, time);
    })
}

屏幕录制 2026-03-23 132454.gif

这个函数就像给红绿灯发了个 “工作通知”:告诉它要亮什么颜色、亮多久,等时间到了,就通知我们 “这个灯的活儿干完了”。

二、多方案实现:让红绿灯按节奏循环亮灯

写法 1:async/await—— 直白的 “指令式” 写法

async/await 是最贴近自然语言的写法,把异步流程写成同步逻辑:

// async/await实现循环亮灯
async function run() {
    while(true) {
        await setColor('红', 3000);  // 等红灯亮完3秒
        await setColor('绿', 2000);  // 等绿灯亮完2秒
        await setColor('黄', 1000);  // 等黄灯亮完1秒
    }
}
run();

await就是 “等一等”,只有前一个灯的任务完成,才会执行下一个,避免 “灯色串岗”。

写法 2:递归 + then 链 ——“接力式” 写法

这是 Promise 原生风格的写法,通过 then 链实现任务接力,递归完成无限循环:

// 递归+then链实现循环亮灯
function run() {
    setColor('红', 3000).then(() => {
        setColor('绿', 2000).then(() => {
            setColor('黄', 1000).then(() => run()); // 黄灯结束后重新循环
        })
    })
}
run();

屏幕录制 2026-03-23 132812.gif

每个灯亮完后,通过 then 方法 “喊” 下一个灯上岗,最后递归调用 run () 重启整个流程。

写法 3:Promise 链 + 定时器 ——“手动计时循环” 写法

如果不想用递归,可以把完整的灯序封装成 Promise 链,再用 setInterval 定时执行:

// 定义完整的灯序执行函数
function lightSequence() {
    return setColor('红', 3000)
        .then(() => setColor('绿', 2000))
        .then(() => setColor('黄', 1000));
}
// 按总时长(6秒)循环执行灯序
setInterval(lightSequence, 6000);
// 立即执行一次,避免首次等待6秒
lightSequence();

屏幕录制 2026-03-23 133350.gif 先定义「红→绿→黄」的完整 Promise 链,再用 setInterval 按总时长(3+2+1=6 秒)循环调用,注意要先手动执行一次,避免页面加载后首次需要等 6 秒才亮灯。

写法 4:生成器函数 + 自动执行 ——“迭代式” 写法

利用 ES6 的生成器函数(Generator)封装灯序逻辑,配合自动执行函数实现循环:

// 生成器函数定义灯序
function* lightGenerator() {
    while(true) {
        yield setColor('红', 3000);
        yield setColor('绿', 2000);
        yield setColor('黄', 1000);
    }
}
// 自动执行生成器的函数
async function runGenerator() {
    const gen = lightGenerator();
    for await (const step of gen) {
        // 自动迭代执行每个灯的任务
    }
}
runGenerator();

屏幕录制 2026-03-23 133814.gif

结语

看似普通的红绿灯,藏着的是 JS 异步编程的核心逻辑:Promise 解决 “等待” 问题,async/await 或递归解决 “顺序 + 循环” 问题。

把抽象的代码和生活场景结合,面试的场景题其实不难 💪

你点的“刷新”是假刷新?前端路由的瞒天过海术

为什么单页应用切换页面时,浏览器没有真正刷新?地址栏变了,页面却没白一下?今天我们来拆穿前端路由的“魔术”——它根本没去服务器要新页面,而是自己偷偷换了内容。看完这篇,你也能实现一个自己的前端路由。

前言

你有没有注意过,现在很多网站(比如知乎、B站、Github)点开一个新页面,地址栏变了,但页面没有那种“白屏-加载-闪现”的过程,而是瞬间切换内容。这就像你走进一家餐厅,菜单上写着“换桌”,你以为换了个房间,结果服务员只是把你桌上的桌布换了。

这就是前端路由干的“好事”。它让页面看起来跳转了,实际上只是JS在背后偷偷换了DOM,地址栏的变化也是骗你的。今天我们就来揭开这个魔术的奥秘,顺便自己写一个简单的路由。

一、什么是前端路由?

传统网站,点击链接会向服务器请求一个新HTML,浏览器刷新整个页面。这叫后端路由

单页应用(SPA)里,所有页面逻辑都在一个HTML里。切换“页面”时,不会请求新HTML,而是JS擦掉旧内容,画上新内容。同时,通过某种手段改变浏览器的地址栏URL,让用户感觉像换了个页面。这就是前端路由

前端路由的实现依赖两个“戏法”:

  • 改变URL但不刷新页面
  • 监听URL变化并渲染对应组件

二、Hash模式:带#号的“假跳转”

早期前端路由用的是hash(也就是URL里#后面的部分)。改变#后的值,不会触发页面刷新,也不会向服务器发请求。浏览器自己会记录历史(前进后退可用)。

// 改变hash
window.location.hash = 'home';

// 监听hash变化
window.addEventListener('hashchange', () => {
  const hash = window.location.hash.slice(1); // 去掉#
  renderPage(hash);
});

比如https://example.com/#/home,你改成#/about,页面不会刷新,但hashchange事件会触发,你可以在回调里根据hash渲染不同内容。

优点:兼容性好,IE也能用。
缺点:URL有个丑陋的#;服务端无法捕获#后面的内容(因为#之后的部分不会发到服务器)。

三、History模式:看起来像真的

HTML5新增了pushStatereplaceState,可以改变URL路径,同样不刷新页面。加上popstate事件监听,就能实现干净的路由(没有#)。

// 改变URL(添加一条历史记录)
history.pushState({ page: 'home' }, 'Home', '/home');

// 替换当前历史记录(不新增)
history.replaceState({ page: 'about' }, 'About', '/about');

// 监听前进后退
window.addEventListener('popstate', (event) => {
  const state = event.state; // pushState时传的数据
  renderPage(location.pathname);
});

优点:URL干净,像真实多页面。
缺点:需要服务端配合——因为刷新页面时,浏览器会按真实路径请求服务器,如果服务器没配置,会404。解决方案:所有路由都返回同一个HTML(即index.html)。

四、手写一个迷你前端路由

我们来实现一个最简单的Hash路由,包含三个“页面”:首页、关于、404。

<nav>
  <a href="#/home">首页</a>
  <a href="#/about">关于</a>
  <a href="#/nothing">不存在</a>
</nav>
<div id="app">内容会变</div>
function renderPage(path) {
  const app = document.getElementById('app');
  if (path === '/home') {
    app.innerHTML = '<h2>🏠 首页</h2><p>欢迎来到我的网站</p>';
  } else if (path === '/about') {
    app.innerHTML = '<h2>📖 关于</h2><p>这是一个前端路由演示</p>';
  } else {
    app.innerHTML = '<h2>❌ 404</h2><p>页面不存在</p>';
  }
}

// 监听hash变化
window.addEventListener('hashchange', () => {
  const hash = window.location.hash.slice(1); // 去掉#
  renderPage(hash || '/home');
});

// 页面加载时执行一次
window.addEventListener('load', () => {
  const hash = window.location.hash.slice(1);
  renderPage(hash || '/home');
});

就这么几行,你已经实现了一个前端路由。当然,实际框架里的路由更复杂(嵌套路由、动态参数、路由守卫等),但核心原理就是监听URL变化 + 渲染对应组件。

五、前端路由与后端路由的区别

特性 后端路由 前端路由
请求方式 每次跳转都请求服务器 不请求服务器(JS切换内容)
刷新页面 会重新下载HTML 会刷新但需要服务端配合(history模式)
首屏加载 只加载当前页面 通常要加载所有JS(可代码分割)
用户体验 有白屏、闪烁 切换流畅
SEO 友好 较差(需SSR或预渲染)

六、常见坑点与解决方案

1. History模式刷新404

配置Nginx将所有路由指向index.html:

location / {
  try_files $uri $uri/ /index.html;
}

2. 路由跳转但页面不滚动

单页切换时,滚动条位置可能保留在上一个页面的位置。需要在路由变化后手动window.scrollTo(0, 0)

3. 动态路由参数

比如/user/:id,你需要从路径中提取id。可以用正则或简单分割:

function matchRoute(path, routePath) {
  const pathParts = path.split('/');
  const routeParts = routePath.split('/');
  if (pathParts.length !== routeParts.length) return null;
  const params = {};
  for (let i = 0; i < pathParts.length; i++) {
    if (routeParts[i].startsWith(':')) {
      params[routeParts[i].slice(1)] = pathParts[i];
    } else if (routeParts[i] !== pathParts[i]) {
      return null;
    }
  }
  return params;
}

七、总结

  • 前端路由让单页应用切换页面时不刷新,体验流畅。
  • Hash模式# + hashchange,兼容性好,但URL丑。
  • History模式pushState + popstate,URL干净,需服务端配合。
  • 原理很简单:监听URL变化 → 根据路径渲染不同内容。
  • 现代框架(React Router、Vue Router)都是在此基础上增强。

下次再看到地址栏变了但页面没白,你就可以自信地说:“哼,不过是在演我。”

如果你喜欢今天的“魔术揭秘”,点个赞让更多人看到。明天我们将聊聊Webpack的Loader和Plugin原理,从零理解构建工具的核心。我们明天见!

本地存储全家桶:从localStorage到IndexedDB,把数据塞进用户浏览器

你有没有想过,为什么刷新页面后,有些网站还能记住你的登录状态?为什么购物车里的商品关掉浏览器再打开还在?今天我们就来聊聊浏览器里的“记忆术”——本地存储。从简单的钥匙串localStorage,到能装下整个图书馆的IndexedDB,总有一款适合你。

前言

想象一下,你每次去网吧上网,都要重新登录所有账号、重新设置主题、重新添加购物车——是不是想砸电脑?还好,浏览器有“记忆功能”。它能在你的电脑里存点东西,下次再来时直接拿出来用。

这个“记忆功能”就是Web存储。今天我们就来盘点一下浏览器提供的几种存储方式:localStorage、sessionStorage、cookie,以及能存视频、存大文件的IndexedDB。看完你就能根据场景选对工具,再也不用担心数据“蒸发”了。

一、localStorage:永不过期的便利贴

localStorage是一个挂在window上的对象,它存的数据没有过期时间,除非你手动清除或者用户清理浏览器缓存,否则会一直待在那里。

基本用法

// 存数据(键值对,值必须是字符串)
localStorage.setItem('username', '张三');
localStorage.setItem('theme', 'dark');

// 取数据
const name = localStorage.getItem('username'); // '张三'

// 删除某条
localStorage.removeItem('theme');

// 全部清空
localStorage.clear();

// 获取存储数量
console.log(localStorage.length);

存对象怎么办?

localStorage只能存字符串,所以对象要先转成JSON:

const user = { name: '张三', age: 18 };
localStorage.setItem('user', JSON.stringify(user));

// 读取时解析
const stored = JSON.parse(localStorage.getItem('user'));

容量限制

大多数浏览器限制5MB~10MB,够存一些配置、用户信息、小量缓存。

特点总结

  • 同步:操作是同步的,会阻塞主线程(但一般很快)。
  • 同源:同一域名下所有页面共享(包括不同标签页)。
  • 永久:除非手动清除。
  • 仅客户端:不会自动发送到服务器。

二、sessionStorage:标签页关闭就消失的临时工

sessionStoragelocalStorage的API一模一样,但生命周期不同:它只存在于当前标签页。关掉标签页,数据就没了。刷新页面还在,但新开标签页(即使是同一个网站)会得到一个新的sessionStorage。

// 用法完全一样
sessionStorage.setItem('tempData', '临时值');

适用场景:表单临时草稿、当前页面的中间状态、不希望跨页面共享的敏感信息。

三、cookie:老前辈,但有点“重”

cookie是最早的浏览器存储机制,但如今除了会话管理(登录态)和少量用户追踪,大部分场景已被localStorage替代。

特点

  • 容量小:每个cookie 4KB 左右。
  • 自动携带:每次HTTP请求都会把cookie发给服务器(增加带宽消耗)。
  • 可设置过期时间。
  • 可标记HttpOnly(禁止JS读取,防XSS)、Secure(仅HTTPS)、SameSite(防CSRF)。
// 设置cookie(繁琐)
document.cookie = "username=张三; expires=Thu, 18 Dec 2026 12:00:00 UTC; path=/";

// 读取cookie(需要自己解析)
console.log(document.cookie);

现在主流做法:用localStorage存非敏感数据,用httpOnly cookie存登录凭证

四、IndexedDB:浏览器里的“小数据库”

如果你要存的东西很大(几百MB),或者需要复杂的查询、索引、事务,那么localStorage就不够用了。这时候请出IndexedDB——一个运行在浏览器里的非关系型数据库。

特点

  • 容量大:通常250MB+,甚至更多(取决于浏览器)。
  • 异步API:基于Promise或回调,不阻塞主线程。
  • 支持索引、游标、事务。
  • 可以存储File、Blob、ArrayBuffer等二进制数据。

快速上手

IndexedDB的API比较原始,不过我们可以封装一下。

// 1. 打开/创建数据库
const request = indexedDB.open('MyDatabase', 1);

request.onupgradeneeded = (event) => {
  const db = event.target.result;
  // 创建一个对象仓库(类似表),指定主键
  const store = db.createObjectStore('users', { keyPath: 'id' });
  // 创建索引,用于快速查询
  store.createIndex('name', 'name', { unique: false });
};

request.onsuccess = (event) => {
  const db = event.target.result;
  console.log('数据库打开成功');
  // 后续增删改查都用这个db对象
};

request.onerror = (event) => {
  console.error('数据库打开失败', event.target.error);
};

增删改查

// 添加数据(在onsuccess里拿到db)
const transaction = db.transaction(['users'], 'readwrite');
const store = transaction.objectStore('users');
const addRequest = store.add({ id: 1, name: '张三', age: 18 });

addRequest.onsuccess = () => console.log('添加成功');
addRequest.onerror = (e) => console.error('添加失败', e.target.error);

// 查询
const getRequest = store.get(1);
getRequest.onsuccess = () => console.log(getRequest.result);

// 更新(使用put,如果存在则覆盖)
store.put({ id: 1, name: '李四', age: 20 });

// 删除
store.delete(1);

使用游标遍历

const range = IDBKeyRange.bound(1, 10); // id从1到10
store.openCursor(range).onsuccess = (e) => {
  const cursor = e.target.result;
  if (cursor) {
    console.log(cursor.value);
    cursor.continue(); // 继续下一个
  }
};

现代封装:localForage

原生IndexedDB API太啰嗦,推荐用localForage这个库,它提供了类似localStorage的简洁API,但背后自动选择IndexedDB、WebSQL或localStorage。

// 使用localForage
import localforage from 'localforage';

await localforage.setItem('user', { name: '张三' });
const user = await localforage.getItem('user');

五、四种存储方式对比

特性 localStorage sessionStorage cookie IndexedDB
容量 5-10MB 5-10MB 4KB 几百MB
生命周期 永久 标签页关闭 可设置过期 永久
跨标签页
异步 同步 同步 同步 异步
自动发到服务器 是(每次请求)
数据类型 字符串 字符串 字符串 任意(结构化克隆)
查询能力 索引、游标

六、选型指南:到底用哪个?

  • 简单键值对,少量数据localStorage,比如用户偏好设置、主题、是否首次访问。
  • 临时数据,只在一个页面用sessionStorage,比如多步骤表单的暂存。
  • 登录凭证httpOnly cookie(安全)配合后端。
  • 大量结构化数据、离线应用IndexedDB,比如邮件客户端、笔记应用、缓存API数据。
  • 需要与后端自动同步cookieAuthorization头(用localStorage存token也行,但要注意XSS)。

七、避坑指南

1. localStorage 的同步阻塞

大量数据存取会阻塞UI,建议不要存超过几MB,或改用IndexedDB。

2. 隐私模式

Safari的隐私模式下,localStorage和IndexedDB可能不可用或容量极低,要写try-catch降级。

3. 序列化问题

localStorage存对象会丢失原型链、函数、Symbol、循环引用。用JSON.stringify前确保数据可序列化。

4. 安全提醒

永远不要把敏感信息(如密码、token)明文存在localStorage,因为任何JS都能读到(XSS攻击)。token建议用httpOnly cookie或短时效+refresh机制。

5. IndexedDB 版本升级

当修改数据库结构时,需要增加版本号,并在onupgradeneeded里处理旧数据迁移,否则会报错。

八、总结:存储就像选工具箱

  • localStorage:日常杂货,随手放。
  • sessionStorage:临时工,关窗走人。
  • cookie:老古董,特殊场合用。
  • IndexedDB:重武器,存大文件、复杂查询。

掌握了这些,你就可以在浏览器里随心所欲地存数据了。明天我们将继续前端工程化的旅程,聊聊Cookie与Session的区别,以及现代认证方案JWT。

如果你觉得今天的存储全家桶够实用,点个赞让更多人看到。我们明天见!

JS 栈与堆内存全解析(含内存泄漏 / 闭包 / GC)

你是否在面试中被问过:JavaScript 的基本类型和引用类型存储在哪里?你是否遇到过页面越用越卡、内存持续飙高,却找不到原因?你是否理解闭包、浅拷贝、内存泄漏和栈堆的底层关系?

在前端开发中,栈(Stack)与堆(Heap)是 JavaScript 内存模型的核心基石,也是面试高频考点、性能优化的关键。很多同学只知其名,不知其理,导致在实际开发中踩坑无数。

今天,我们就结合你的核心知识点,从零到一吃透 JS 栈堆内存、垃圾回收、内存泄漏、闭包 等硬核知识点,全文逻辑闭环、内容详实,无论是面试还是实战优化都直接能用。


一、开篇三问:你真的了解 JS 内存吗?

在深入栈堆之前,我们先抛出三个灵魂问题,带着问题学习更有方向:

  1. 为什么 let a = 1let a = {} 赋值、拷贝的表现完全不同?
  2. 为什么函数执行完,局部变量就消失了,而闭包变量能一直保留?
  3. 为什么项目跑久了会卡顿?内存泄漏到底和栈堆有什么关系?

这三个问题的答案,全部藏在 JavaScript 的栈堆内存模型 里。接下来我们逐层拆解,从内存分类、存储规则、管理方式,到垃圾回收、内存泄漏,一次性讲透。


二、第一部分:JavaScript 内存模型 —— 栈与堆

在 JavaScript 中,引擎会把内存划分为 ** 栈内存(Stack)堆内存(Heap)** 两部分,二者分工明确、各司其职,共同支撑 JS 代码的运行。

1. 栈内存(Stack):系统自动管理的高速内存

栈内存是 JavaScript 中执行代码、存储简单数据的核心区域,它的管理方式完全由系统自动完成,开发者无需手动干预。

栈内存存储什么?

  • 执行上下文(全局上下文、函数执行上下文)
  • 基本数据类型(Number、String、Boolean、Null、Undefined、Symbol、BigInt)
  • 函数调用记录(调用栈)
  • 引用类型的内存地址(指针)

栈内存核心特点

  1. 操作速度极快栈是 CPU 最友好的内存结构,读写只需要移动指针,效率远高于堆内存。
  2. 内存空间连续栈内存是一块连续的存储空间,遵循 LIFO(后进先出) 原则,和数据结构中的栈完全一致。
  3. 系统自动分配与释放函数执行时入栈,执行完毕后,栈内数据自动出栈销毁,不需要垃圾回收机制(GC) 参与。
  4. 存储空间小、固定大小栈内存大小有限,不适合存储大型、复杂的数据。

一句话总结栈内存:小而快、自动管、存简单值 / 地址


2. 堆内存(Heap):GC 管理的动态内存

堆内存用于存储复杂、占用空间大的数据,它的管理方式和栈内存完全不同。

堆内存存储什么?

  • 对象(Object)
  • 数组(Array)
  • 函数(函数的调用逻辑存在栈中,函数体本身存在堆中)
  • 所有引用类型的真实数据

堆内存核心特点

  1. 内存空间不连续堆内存是散乱分配的,动态申请空间,会产生内存碎片
  2. 操作速度较慢分配和回收都需要计算,读写效率低于栈。
  3. 手动 / GC 自动管理底层语言(C/C++)需要 new 申请、delete 释放;JavaScript 不需要手动操作,由 GC(垃圾回收机制) 自动管理。
  4. 存储空间大、动态大小可以存储任意大小的复杂数据。

一句话总结堆内存:大而慢、GC 管、存真实对象数据


3. 栈与堆的协作关系

在 JavaScript 中,变量本身在栈中,而对象实际存在堆中,栈里存的是指向堆的引用地址。

举个最经典的例子:

// 基本类型:直接存在栈内存
let num = 100;
let str = "前端";

// 引用类型:栈存地址,堆存真实数据
let obj = { name: "掘金" };
let arr = [1, 2, 3];

执行这段代码时:

  • 栈内存:存储 numstr 的值,存储 objarr堆内存引用地址
  • 堆内存:存储 {name:"掘金"}[1,2,3] 的真实数据

这就是 JS 内存最核心的规则:栈里存的是简单类型值 或 引用地址;堆里存的是对象的真实数据。


4. 栈与堆性能差异深度对比

  • 栈快(只需要挪指针),堆慢
  • 栈连续内存(LIFO,无需 GC),堆非连续(随机分配,需要 GC)
  • 闭包,本该在栈释放的数据,被引用到了堆中

从生命周期来看:栈中的数据随着函数执行结束自动释放,而堆中的数据只要引用存在就不会被回收。


三、第二部分:数据结构中的栈与堆

除了内存模型,栈和堆也是数据结构中的常客,面试中经常会把「内存栈堆」和「数据结构栈堆」放在一起问,我们必须区分清楚。

1. 数据结构中的栈

栈是一种线性数据结构,严格遵循 LIFO(Last In First Out,后进先出) 原则。

  • 只能从一端添加 / 删除数据(栈顶)
  • 经典应用:函数调用栈、括号匹配、浏览器后退功能

它和内存中的栈内存规则完全一致,这也是栈内存得名的原因。

2. 数据结构中的堆

堆是一种非线性的树形数据结构,和内存中的堆完全不是一个概念!

数据结构中的堆分为两种:

  • 大顶堆:父节点值 ≥ 子节点值
  • 小顶堆:父节点值 ≤ 子节点值

堆常用于:优先队列、堆排序、TOP K 问题。

⚠️ 重要区分:

  • 内存堆:存储引用类型,GC 管理
  • 数据结构堆:排序、优先队列

面试中如果同时问到,一定要清晰区分二者,不要混淆。


四、第三部分:JS 垃圾回收机制(GC)—— 内存的 “清洁工”

堆内存需要垃圾回收机制来释放空间,GC 就是 JS 引擎的内存清洁工,负责把「不再使用的对象」清理掉,释放内存。

1. 对象可回收的核心标准:是否可达

GC 判断一个对象是否能被回收,只有一个标准:这个对象是否还能被访问到?(是否可达)

只要对象无法通过任何方式被访问,GC 就会标记它,并在合适时机回收内存。

我们用四个经典案例,彻底讲透「对象可达性」:

案例 1:局部对象 —— 自动回收

function fn(){
    let obj={a:1}  //会被回收
}
fn();

函数执行完毕,执行上下文出栈,obj 变量销毁,堆中对象无引用 → 不可达 → 被回收。

案例 2:全局引用 —— 不会回收

let globalObj;
function fn(){
    let obj={a:1}   //对象引用 n=1  不会释放
    globalObj=obj;  //对象引用 n=1+1
}
fn(); //n-1=1

函数执行完,局部变量 obj 销毁,但 globalObj 仍在引用,对象始终可达 → 不会被回收。

案例 3:循环引用

let a={};
let b={};
a.x=b;
b.y=a;
// 循环引用

a=null;
b=null;

老式引用计数算法会因计数不为 0 无法回收,造成泄漏;现代标记清除算法会判断不可达,正常回收。

案例 4:闭包

function outer(){
    let obj={a:1}
    return function inner(){
        console.log(obj)
    }
}
let fn=outer();

内层函数引用外层变量,导致 obj 一直可达,不会被回收。这就是闭包的本质:栈上本该释放的变量,被引用到了堆中,延长了生命周期。


2. 垃圾回收算法

(1)引用计数

  • 被引用一次 +1,取消引用 -1
  • 计数为 0 则回收
  • 致命缺陷:循环引用无法回收
  • 现已基本废弃

(2)标记清除(V8 主流)

  • 从根对象(window/global)开始遍历
  • 给可达对象打标记,未标记对象清除
  • 解决循环引用问题
  • 缺点:产生内存碎片

通俗理解:打扫卫生,哪些要断舍离呢?贴个标签,没贴的就扔掉回收。


五、第四部分:内存泄漏 —— 前端的 “隐形杀手”

内存泄漏是前端应用中的隐形杀手。本该被回收的内存,因被意外引用而无法释放,导致占用持续增长,最终引发页面卡顿甚至崩溃。

它不会立即导致应用崩溃,而是像慢性病一样,随着用户使用时间的延长,逐渐吞噬系统资源,最终导致卡顿、延迟,甚至浏览器标签页崩溃。

下面我们逐一梳理最常见的 9 种内存泄漏场景,并给出解决方案。

1. 定时器未清理

React / Vue 组件卸载了,但定时器没有取消。定时器内部函数引用了外部变量,变量被长期引用,无法释放。

解决方案:组件卸载时清除定时器。

2. 事件监听未移除

给 window、document、DOM 绑定事件后,组件销毁未解绑,回调函数长期驻留内存。

3. DOM 引用未释放

let el=document.getElementById('app')
document.body.removeChild(el);
// DOM 已删,但引用还在
el=null; // 必须手动切断

4. 全局变量 / 意外挂载到全局

  • var 声明自动挂载 window
  • 未声明直接赋值自动全局window 是根对象,永远可达 → 永不回收。

5. 闭包导致的内存泄露

不必要的长生命周期闭包,会让内部引用对象一直存活。

6. Map/Set 使用不当

Map/Set 是强引用,即使 key 对象设为 null,Map 依然持有引用,无法回收。

解决方案:使用 WeakMap、WeakSet,它们是弱引用,key 对象被回收时会自动移除对应项,不会造成泄漏。

7. 订阅发布者模式

只订阅不取消订阅,回调函数长期引用外部变量,造成泄漏。

8. Promise 一直不结束

Promise 永久 pending,内部持有大量引用无法释放。

9. 请求未中止,组件已卸载

请求发送后,组件提前卸载,回调仍持有引用。解决方案:使用 AbortController 中止请求。


六、第五部分:栈堆模型对实际开发的深层影响

栈堆不只是面试题,它直接决定了你代码的行为、性能和 Bug 来源。

1. 影响浅拷贝与深拷贝

  • 基本类型赋值:拷贝栈值,相互独立
  • 引用类型赋值:拷贝地址,共享堆数据

浅拷贝只复制一层地址,深拷贝重新开辟堆内存,完整复制所有结构。理解栈堆,你就彻底理解深浅拷贝的本质区别。

2. 影响闭包原理

闭包的核心就是:栈上的执行上下文销毁了,但变量被堆中的函数引用,因此保留在内存中。

3. 影响内存泄漏

绝大多数泄漏,本质都是:堆对象被意外长期引用,GC 无法回收。

4. 影响页面性能

  • 栈内存几乎无性能压力
  • 堆内存过多、频繁 GC → 主线程阻塞 → 页面卡顿

七、第六部分:面试满分回答模板(可直接背诵)

如果面试官问:说说 JS 中的栈和堆?

你可以这样回答:

在 JavaScript 中,内存分为栈内存和堆内存。栈主要存储执行上下文、基本类型值、以及引用类型的地址,由系统自动管理,内存连续、后进先出、速度极快,函数执行完自动释放。堆存储对象、数组、函数等复杂数据,内存不连续,需要垃圾回收机制管理,速度相对较慢。

变量存在栈中,对象真实数据存在堆中,栈保存堆的引用地址。垃圾回收主要通过标记清除算法,判断对象是否可达来决定是否回收。

内存泄漏通常是因为对象被意外长期引用,比如未清理定时器、事件监听、DOM 引用、循环引用、不当闭包、强引用 Map 等。实际开发中可以通过及时清理监听、使用 WeakMap、避免冗余闭包来避免泄漏。

这一段覆盖所有要点,逻辑清晰,面试官直接给高分。


八、总结

栈和堆核心区别在于存储内容和管理方式

  • :执行上下文、基本类型、引用地址,系统自动管理,连续内存、LIFO、速度快、无碎片。
  • :对象、数组、函数真实数据,GC 管理,非连续、速度较慢、有碎片。

在 JS 中:变量本身在栈中,对象实际数据在堆中,栈存堆地址。 栈随函数结束自动释放;堆只有无引用时才被 GC 回收。

栈堆模型直接影响:深浅拷贝、闭包行为、垃圾回收、内存泄漏、页面性能。

理解栈堆,才算真正理解 JavaScript 的运行底层。

⏰前端周刊第 459 期v2026.4.3

📢 宣言每周更新国外论坛的前端热门文章,推荐大家阅读/翻译,紧跟时事,掌握前端技术动态,也为写作或突破新领域提供灵感~

欢迎大家访问:github.com/TUARAN/fron… 顺手点个 ⭐ star 支持,是我们持续输出的续航电池🔋✨!

在线网址:frontendweekly.cn/

前端周刊封面


💬 推荐语

本期更像一次很清晰的信号汇总:浏览器平台继续补齐“过去需要框架和脚本硬扛”的能力,而工程现实也在继续逼我们重新看待安全、可维护性与交互边界。

一边是平台层继续前进。Chrome 147 带来 element-scoped view transitions,setHTML() 这样的安全 API 开始走向更实用的位置,Shadow DOM 的 delegatesFocus、Anchor Positioning、自动进入视频画中画、文本缩放支持等话题,也都在说明 Web 平台正在一点点把历史上的“边缘痛点”收回来。

另一边,工程和体验问题并没有因为平台变强而自动消失。DevTools 继续强化调试能力,Three.js Conf 网站与 Codrops 的动效案例在拉高体验上限,Deno Sandbox、开发者工具和 AI/平台协作方向的文章,则提醒我们:现代前端已经不只是框架选型,而是在同时处理能力暴涨、交互复杂化与安全边界重构。

如果把这期压缩成一句话,那就是:前端的下一阶段,拼的不是“会不会追新”,而是能不能准确判断哪些复杂度应该交给平台,哪些还得自己治理。


🗂 本期精选目录

🧭 Web 开发

🛠 工具与工程

🎨 CSS 与交互

✨ 创意开发


结语

setHTML()、element-scoped view transitions、delegatesFocus 到 Anchor Positioning,这期最值得记住的不是某一个孤立新特性,而是一个越来越明确的趋势:Web 平台正在主动接管更多曾经只能靠框架、脚本和经验兜底的问题。

但平台变强并不等于工程变简单。安全边界、组件可用性、调试复杂度、交互组织方式,依旧需要团队做出判断。真正有价值的前端工作,越来越不是追着每个新词跑,而是看清哪些能力已经成熟到值得纳入日常体系,哪些地方又仍然必须谨慎设计。

从 Next.js 迁移到 Pareto:哪些变了,哪些没变

你熟悉 Next.js,熟悉文件路由、布局、SSR。你大概也熟悉那些痛点:Server Components vs Client Components,满屏的 "use client",莫名其妙的 hydration 错误,还有你一行业务代码没写就已经 233 KB 的客户端包。

Pareto 提供同样的 SSR 模式——但没有这些复杂性。标准 React 组件,Vite 替代 Webpack/Turbopack,客户端包只有 62 KB。这篇文章详细对比从 Next.js 切到 Pareto 时,什么变了,什么不变。

心智模型的转变

Next.js(App Router): 每个组件默认是 Server Component。想用 useState?加 "use client"。数据获取通过 async server component 或者 generateMetadata。你无时无刻不在思考 server/client 边界。

Pareto: 每个组件都是普通 React 组件,同时运行在服务端和客户端。数据获取在 loader.ts 文件中完成——借鉴了 Remix 的模式。没有 "use client" 指令,因为根本不存在 Server Component / Client Component 的划分。

Next.js 心智模型:  "这是 Server Component 还是 Client Component?"
Pareto 心智模型:   "这是数据还是 UI?"

路由:几乎一模一样

如果你熟悉 Next.js App Router 的约定,Pareto 的路由会立刻上手:

Next.js Pareto 用途
page.tsx page.tsx 路由组件
layout.tsx layout.tsx 包裹布局
loader.ts 服务端数据获取
loading.tsx Suspense + <Await> 加载状态
error.tsx ParetoErrorBoundary 错误处理
not-found.tsx not-found.tsx 404 页面
route.ts route.ts API 端点
generateMetadata head.tsx Meta 标签

最大的区别:Pareto 用独立的 loader.ts 文件做数据获取,而不是把页面组件变成 async。

数据获取:loader 替代 async 组件

Next.js(App Router):

// app/dashboard/page.tsx(server component)
export default async function Dashboard() {
  const stats = await db.getStats()
  return <h1>{stats.total} users</h1>
}

Pareto:

// app/dashboard/loader.ts
import type { LoaderContext } from '@paretojs/core'

export function loader(ctx: LoaderContext) {
  return { stats: db.getStats() }
}

// app/dashboard/page.tsx
import { useLoaderData } from '@paretojs/core'

export default function Dashboard() {
  const { stats } = useLoaderData<{ stats: { total: number } }>()
  return <h1>{stats.total} users</h1>
}

两个文件替代一个,但分离是有意为之:数据获取是显式的、可测试的,永远不与渲染逻辑混在一起。组件是标准 React——没有 async,没有 await,没有 server-only 限制。

流式渲染:defer() 替代 Suspense 体操

Next.js: 流式渲染需要把页面拆分成 server 和 client 组件,协调 loading.tsx 边界,理解哪些组件会阻塞首次渲染。

Pareto: 在 loader 中调用 defer(),用 <Await> 包裹慢数据。搞定。

// app/dashboard/loader.ts
import { defer } from '@paretojs/core'

export async function loader() {
  const userCount = await getUserCount()  // 先解析快数据

  return defer({
    userCount,                             // 已解析——立即发送
    activityFeed: getActivityFeed(),       // 慢——流式传输
    analytics: getAnalytics(),             // 更慢——稍后流式传输
  })
}

// app/dashboard/page.tsx
import { useLoaderData, Await } from '@paretojs/core'

export default function Dashboard() {
  const { userCount, activityFeed, analytics } = useLoaderData()

  return (
    <div>
      <h1>{userCount} users</h1>

      <Await resolve={activityFeed} fallback={<Skeleton />}>
        {(feed) => <ActivityList items={feed} />}
      </Await>

      <Await resolve={analytics} fallback={<ChartSkeleton />}>
        {(data) => <AnalyticsChart data={data} />}
      </Await>
    </div>
  )
}

每个 <Await> 创建独立的 Suspense 边界。快数据立即渲染,慢数据逐步流入。初始 SSR 和客户端导航行为一致(Pareto 4.0 通过 NDJSON 流式传输实现)。

Head 管理:React 组件,不是配置对象

Next.js:

export async function generateMetadata({ params }) {
  const post = await getPost(params.id)
  return { title: post.title, description: post.excerpt }
}

Pareto:

// app/blog/[id]/head.tsx
export default function Head({ loaderData }: { loaderData: { post: Post } }) {
  return (
    <>
      <title>{loaderData.post.title}</title>
      <meta name="description" content={loaderData.post.excerpt} />
      <meta property="og:title" content={loaderData.post.title} />
    </>
  )
}

就是一个 React 组件。可以用条件逻辑、组合共享组件、渲染任何合法的 <head> 内容。Head 组件从根布局到页面依次合并——最深层路由的重复标签优先。

状态管理:内置,不是外挂

Next.js 对状态管理没有意见。你自己装 Redux、Zustand、Jotai,然后自己搞定 SSR hydration。

Pareto 内置 defineStore(),集成 Immer:

import { defineStore } from '@paretojs/core/store'

const { useStore, getState, setState } = defineStore((set) => ({
  items: [] as CartItem[],
  total: 0,
  addItem: (item: CartItem) => set((d) => {
    d.items.push(item)
    d.total += item.price
  }),
}))

SSR hydration 全自动。服务端定义的状态自动序列化并在客户端恢复,不需要任何手动的 dehydrate / rehydrate 样板代码。

配置:一个文件

Next.js: next.config.js 框架配置 + 单独的 Webpack/Turbopack 定制 + 可能还有 middleware.ts + 环境变量约定。

Pareto: 一个 pareto.config.ts

import type { ParetoConfig } from '@paretojs/core'

const config: ParetoConfig = {
  configureVite(config) {
    // 标准 Vite 配置——你的插件直接能用
    return config
  },
  configureServer(app) {
    // 标准 Express app——加任何中间件
    app.use(cors())
  },
}

export default config

没有框架魔法。底层就是 Vite 和 Express,完全可控。

性能差距

我们在 CI 中跑自动化基准测试,在相同硬件上对比 Pareto 和 Next.js:

  • 数据加载吞吐量: Pareto 2,733 req/s vs Next.js 293 req/s(9.3 倍
  • 流式 SSR 容量: Pareto 2,022 req/s vs Next.js 310 req/s(6.5 倍
  • 客户端 JS 包: 62 KB vs 233 KB(小 73%

换算成基础设施:一个需要 2,000 req/s 的页面,Pareto 需要 1 台服务器,Next.js 需要 6 台。完整基准测试数据:paretojs.tech/blog/benchm…

你会放弃什么

公开透明很重要。Pareto 没有的东西:

  • Server Components — 没有 RSC,没有 "use client"。这是设计选择:loader 模式更简单,覆盖 95% 的场景。
  • 图片优化 — 没有 <Image> 组件。用标准 <img> + CDN。
  • ISR / 静态生成 — Pareto 只做 SSR。没有构建时渲染。
  • 中间件 — 没有 Edge Middleware。用 configureServer() 中的 Express 中间件替代。
  • Vercel 集成 — 没有一键部署。你部署的是标准 Node.js 服务器。
  • 生态规模 — 更小的社区,更少的示例。你是早期用户。

如果你在做内容驱动的营销站需要 ISR,Next.js 仍然是对的选择。如果你在做数据驱动的应用、性能和简洁性很重要,Pareto 值得切换。

迁移清单

  1. npx create-pareto@latest my-app — 创建新项目
  2. 移动路由文件 — 目录结构几乎一样
  3. 把 async server component 拆分为 loader.ts + 标准组件
  4. 删掉 "use client" 指令 — 不需要了
  5. generateMetadata 迁移到 head.tsx 组件
  6. loading.tsx 替换为 defer() + <Await> 流式渲染
  7. next/link 换成 @paretojs/coreLink
  8. 把 Webpack 配置迁移到 pareto.config.tsconfigureVite()
  9. 作为标准 Node.js 服务器部署
npx create-pareto@latest my-app
cd my-app && npm install && npm run dev

Pareto — 轻量级流式 React SSR 框架 | 文档

告别页面卡顿:生动解析 JavaScript 防抖与节流

告别页面卡顿:生动解析 JavaScript 防抖与节流

在前端开发的“战场”上,我们常常会遇到一些“话痨”事件:用户疯狂点击按钮、快速输入搜索词、或者像拉面条一样不停拖动窗口大小。如果对这些高频事件来者不拒,浏览器很快就会因为处理不过来而“罢工”,导致页面卡顿甚至崩溃。

为了解决这个问题,我们需要两位“交通指挥官”:防抖(Debounce)节流(Throttle)

今天,我们就结合一段经典的实战代码,来看看这两位指挥官是如何维持秩序,让页面性能稳如泰山的。


️ 核心代码:两位指挥官的“真身”

首先,让我们直面你提供的这段核心代码。这不仅仅是几行 JavaScript,这是控制事件频率的“宪法”。

我们将以这个通用的 ajax 请求函数作为被管理的对象:

function ajax(content) {
    console.log('ajax request', content);
}

️ 防抖: “等你想好了再告诉我”

防抖(Debounce)的核心逻辑是:“不管你怎么触发,我只在最后一次操作结束后的 N 毫秒执行。”

这就好比电梯关门:电梯门打开后,只要还有人陆续进来(触发事件),门就会一直开着,计时器重置。只有当一段时间(比如 5 秒)没人进出了,门才会真正关上(执行函数)。

让我们看看代码是如何实现这一逻辑的:

function debounce(fn, delay) {
    var id; // 闭包中的自由变量,用于存储定时器ID
    return function(args) {
        if(id) clearTimeout(id); // 核心:如果之前有定时器,立马清除(重置计时)
        var that = this;
        
        id = setTimeout(function() {
            fn.call(that, args) // 延迟执行真正的函数
        }, delay);
    }
}
  • 闭包的妙用var id 被包裹在闭包中,这意味着每次触发事件时,我们都能访问到同一个定时器变量。
  • 重置机制clearTimeout(id) 是防抖的灵魂。用户在输入框里每敲一个键,之前的计时就被打断,重新开始倒数。
  • this 的指向:代码中特意保存了 var that = this,并在 setTimeout 中使用 fn.call(that, args)。这是为了防止定时器执行时 this 意外指向 window,确保上下文环境正确。

节流: “不管多急,请按排队顺序来”

节流(Throttle)的核心逻辑是:“不管你怎么触发,我每隔 N 毫秒只执行一次。”

这就像水龙头滴水:无论你水龙头拧得有多快多猛,水滴只能按照固定的频率一滴一滴往下落。或者想象一下机枪射击,扣住扳机不放,子弹也是按射速一颗颗射出,而不是一瞬间把弹夹全打光。

代码实现稍微复杂一点,它结合了“时间戳”和“定时器”的双重保险(混合版节流):

function throttle(fn, delay) {
    let last, // 记录上次执行的时间戳
        deferTimer; // 定时器
    return function(args) {
        let that = this; 
        let _args = arguments; 
        let now = + new Date(); // 获取当前时间戳
        
        // 如果上次执行过,且当前时间还没到下次执行的时间点(在冷却期内)
        if(last && now < last + delay) {
            clearTimeout(deferTimer);
            // 设置一个定时器,确保在冷却期结束后至少执行一次(这是混合版的优势)
            deferTimer = setTimeout(function() {
                last = now;
                fn.apply(that, _args);
            }, delay);

        } else { 
            // 第一次触发,或者已经过了冷却期,立即执行
            last = now;
            fn.apply(that, _args);
        }
    }        
}
  • 时间戳判断now < last + delay 用来判断是否处于“冷却时间”内。
  • 混合策略:这段代码非常精妙。如果在冷却期内,它会设置一个 deferTimer。这意味着,如果用户一直触发事件,函数不仅会立即执行一次(else 分支),还会在停止触发后的 delay 时间后再执行一次(if 分支里的 setTimeout)。这保证了操作的开头结尾都不会被遗漏。

️ 实战演练:三个输入框的较量

为了直观地展示效果,代码中设置了三个输入框,分别对应“无限制”、“防抖”和“节流”三种状态:

const inputa = document.getElementById('undebounce'); // 裸奔的输入框
const inputb = document.getElementById('debounce');   // 穿了防抖铠甲
const inputc = document.getElementById('throttle');   // 装了节流阀门

let debounceAjax = debounce(ajax, 500); // 500ms 防抖
let throttleAjax = throttle(ajax, 1000); // 1000ms 节流

// 1. 无限制:用户每敲一个字,控制台就打印一次。如果敲得快,请求会堆积。
inputa.addEventListener('keyup', function(e) {
    ajax(e.target.value) 
})

// 2. 防抖:用户快速输入 "Hello",控制台只会打印一次 "Hello"。
// 只有当用户停下手超过 500ms,请求才会发送。
inputb.addEventListener('keyup', function(e) {
    debounceAjax(e.target.value) 
})

// 3. 节流:用户快速输入。
// 第一次按键立即打印。
// 接下来 1 秒内的按键会被忽略,或者在停止 1 秒后打印最后一次。
inputc.addEventListener('keyup', function(e) {
    throttleAjax(e.target.value);
})

总结:何时请哪位指挥官?

特性 防抖 节流
核心逻辑 最后一次说了算 固定频率执行
生活比喻 电梯关门、核弹发射按钮 水龙头滴水、机关枪射击
适用场景 搜索框输入(等用户输完再搜)、窗口 resize(等拖完再计算布局)、表单验证 滚动加载(scroll 事件,每隔一段距离加载一次)、按钮点击(防止重复提交)、鼠标移动(mousemove)
代码特征 clearTimeout 是核心 Date.now()setTimeout 周期性执行

一句话口诀: 如果**“等用户停下来再做”,请用防抖**; 如果**“不管用户多快,我要按节奏来”,请用节流**。

掌握这两段代码,你就掌握了前端性能优化的半壁江山! 这篇解析是否清晰地展示了防抖与节流的区别?如果需要,我可以为你补充这两个函数的定时器版节流实现,或者整理一份面试中常见的防抖节流考点,你需要吗?

🔍 React 面试官眼中的“秘密武器”:深度剖析 useRef

🔍 React 面试官眼中的“秘密武器”:深度剖析 useRef

在 React 的面试战场上,useRef 绝对是那个看似不起眼,实则能决定你能否进入下一轮的“关键先生”。很多候选人能熟练使用 useStateuseEffect,但当被问及“如何在不触发重渲染的情况下保存数据”或“为什么我的定时器关不掉”时,往往哑口无言。

今天,我们就通过两段典型的实战代码,揭开 useRef 的神秘面纱,让你在面试中不仅能写出代码,更能讲出原理。

📌 核心考点:useRef 是什么?

在深入代码前,你需要向面试官传达两个核心概念:

  1. 它是一个“盒子”useRef 返回一个普通的 JavaScript 对象,这个对象在组件的整个生命周期内保持引用稳定(始终是同一个对象)。
  2. 它是“非响应式”的:修改 .current 属性不会触发组件的重新渲染。这是它与 useState 最根本的区别。

💡 场景一:DOM 的直接掌控者

面试官提问: “如何在组件挂载后,让输入框自动获得焦点?”

这通常是考察 useRef 最经典的入门题。我们来看第一段代码:

import { useEffect, useRef, useState } from "react"

export default function App() {
  const [count, setCount] = useState(0);
  console.log('变了'); // 每次count变化,组件重渲染,这句都会执行
  
  const inputRef = useRef(null); // 创建一个ref容器
  
  useEffect(() => {
    // 在这里操作DOM
    inputRef.current.focus(); // 让input获得焦点
    console.log('222', inputRef.current); // 打印真实的DOM节点
  }, [])
  
  // 注意看这里:在渲染函数体中,inputRef.current 是 null
  console.log('111', inputRef.current); 
  
  return (
    <>
      <input ref={inputRef}/> {/* 将ref绑定到DOM上 */}
      <button type="button" onClick={() => setCount(count+1)}>count ++</button>
    </>
  )
}

面试解析要点:

  1. 同步与异步:请留意代码中的 console.log
    • 111:在函数体直接打印,此时 React 还在“画”界面,DOM 节点尚未生成,所以值为 null
    • 222:在 useEffect 中打印,此时 React 已经把界面“挂载”到页面上,inputRef.current 已经被赋值为真实的 DOM 元素。
  2. 执行时机useEffect 在浏览器完成渲染后执行,这保证了我们操作的 DOM 是真实存在的。
  3. 价值体现:这是 useRef 最直观的用途——打破 React 的虚拟 DOM 抽象层,直接操作真实 DOM

⏳ 场景二:跨越渲染的“持久化”存储

面试官提问: “React 函数组件每次渲染都会重新执行函数,如果我在一个定时器里需要保存状态,或者想在点击按钮时清除上一次的定时器,该怎么办?”

这时候,如果只用普通变量,每次重渲染变量都会被重置;如果用 useState,清除定时器的逻辑会变得复杂且容易出错。第二段代码展示了 useRef 的高级用法:

import { useRef, useState, useEffect } from 'react';

export default function App() {
    let intervalId = useRef(null); // 用ref来保存定时器ID
    const [count, setCount] = useState(0);
    
    function start() {
        // 即使组件多次渲染,intervalId 这个“盒子”始终是同一个
        intervalId.current = setInterval(() => {
            console.log('tick~~~');
        }, 1000);
        console.log(intervalId); // 打印 { current: 123 }
    }
    
    function stop() {
        // 从“盒子”里取出ID并清除
        clearInterval(intervalId.current);
    }

    // 仅用于演示:当count变化时,查看ref的值
    useEffect(() => {
        console.log(intervalId.current); // 只要定时器开着,这里会一直打印ID
    }, [count])
    
    return (
        <>
           <button onClick={start}>开始</button>
           <button onClick={stop}>停止</button>
           <button type="button" onClick={() => setCount(count+1)}>count ++</button>
        </>
    )
}

面试解析要点:

  1. 闭包陷阱的解药:在 start 函数中,我们将 setInterval 返回的 ID 存入了 intervalId.current。无论组件因为 count 变化重渲染多少次,intervalId 对象本身不会变,变的只是它里面的 current 值。
  2. 清理资源stop 函数能够准确获取到最新的定时器 ID 并清除它。如果不用 useRef,而用普通变量,stop 函数将无法访问到 start 函数内部的变量(作用域隔离)。
  3. 非响应式优势:我们将定时器 ID 存入 ref,并不希望它触发页面刷新。useRef 完美地充当了一个“默默奉献的存储柜”角色。

📝 总结:面试官想听到什么?

当被问及 useRef 时,请务必构建以下回答框架:

  1. 定义:它是用来创建一个在组件生命周期内持久化且引用稳定的对象。
  2. 两大用途
    • 访问 DOM:通过 ref 属性附加到元素上。
    • 存储可变值:存储定时器 ID、上一次的 props、第三方库实例等不需要触发视图更新的数据。
  3. useState 的区别ref 的变化是同步的且不触发重渲染;state 的变化是异步的且一定会触发重渲染。
  4. 避坑指南:不要试图用 useRef 替代状态管理,因为它的变化 React “看不见”,UI 不会自动更新。

掌握这些,你就能在面试中自信地告诉面试官:useRef** 不仅仅是一个获取 DOM 的工具,它是连接函数组件渲染周期与可变状态的桥梁。**

MutationObserver:DOM界的“卧底”,暗中观察每个风吹草动

你想知道页面上的某个元素什么时候被偷偷改了吗?比如有个熊孩子脚本悄悄改了你的广告位,或者某个懒加载图片终于加载完了?今天我们就来请一位“卧底”——MutationObserver,让它24小时盯着DOM树,任何变化都逃不过它的眼睛。

前言

假设你开了一家便利店,店里装了监控。你想知道:什么时候有人进来?什么时候货架上的商品被拿走了?什么时候价格标签被换了?普通的监控只能录像,但你需要的是“智能警报”——一有变化就通知你。

这就是MutationObserver的活。它是浏览器提供的一个API,专门用来监听DOM树的变化:节点增删、属性修改、文本内容改变……统统能抓到。而且它不会像setInterval那样一直轮询,性能好得多。

一、MutationObserver是啥?

MutationObserver是一个构造函数,用来创建一个观察者对象。你可以给它指定一个回调函数,然后让它去“盯”某个DOM节点。一旦这个节点或它的子孙节点发生变化,回调函数就会被触发。

// 创建一个观察者实例,传入回调
const observer = new MutationObserver((mutationsList, observer) => {
  for (let mutation of mutationsList) {
    console.log(mutation.type, '发生了变化');
  }
});

// 指定要观察的节点
const targetNode = document.getElementById('watch-me');

// 开始观察
observer.observe(targetNode, {
  attributes: true,    // 观察属性变化
  childList: true,     // 观察子节点增删
  subtree: true,       // 观察所有后代节点
  characterData: true  // 观察文本内容变化
});

// 某天不想观察了
// observer.disconnect();

二、能观察到哪些变化?

配置选项决定了你关心哪些“风吹草动”:

  • attributes:属性变了(比如classstylesrc被改)
  • childList:子节点被增删(添加或删除元素、文本节点)
  • characterData:文本节点的内容变了
  • subtree:是否监听后代节点(默认false,只监听目标节点)
  • attributeFilter:只监听特定属性,比如['class', 'src']
  • attributeOldValue:是否记录旧属性值
  • characterDataOldValue:是否记录旧文本值

三、实战:监听广告位有没有被篡改

很多网站会在页面上放广告,但有些恶意脚本会偷偷把广告位换成自己的内容。用MutationObserver可以第一时间发现并报警。

<div id="ad-container">
  <img src="real-ad.jpg" alt="官方广告">
</div>
const adContainer = document.getElementById('ad-container');

const observer = new MutationObserver((mutations) => {
  mutations.forEach((mutation) => {
    if (mutation.type === 'childList') {
      // 子节点被改了
      console.warn('⚠️ 广告位内容被篡改!');
      // 可以上报服务器,或者恢复内容
    } else if (mutation.type === 'attributes' && mutation.attributeName === 'src') {
      console.warn('⚠️ 广告图片被替换了!');
    }
  });
});

observer.observe(adContainer, {
  childList: true,
  subtree: true,
  attributes: true,
  attributeFilter: ['src', 'href']
});

四、实战:监听输入框内容变化(代替input事件?)

input事件已经能监听输入框变化,但MutationObserver可以监听更底层的文本节点变化,比如通过JS直接修改.valueinput事件可能不触发,但MutationObserver可以。

<input id="username" type="text">
const input = document.getElementById('username');

const observer = new MutationObserver((mutations) => {
  mutations.forEach((mutation) => {
    if (mutation.type === 'attributes' && mutation.attributeName === 'value') {
      console.log('输入框的值被改了,新值:', input.value);
    }
  });
});

observer.observe(input, {
  attributes: true,
  attributeFilter: ['value']
});

注意:这种方式监听value属性变化,只对通过JS设置.value有效,用户手动输入不会触发(因为用户输入不改变value属性,而是改变元素的defaultValue和内部状态)。所以实际中监听输入框还是input事件更合适。这里只是演示能力。

五、实战:监听动态加载的图片,做懒加载

很多懒加载库用IntersectionObserver,但如果你想知道图片什么时候被添加到DOM,可以用MutationObserver。

const observer = new MutationObserver((mutations) => {
  mutations.forEach((mutation) => {
    mutation.addedNodes.forEach((node) => {
      if (node.nodeType === 1 && node.tagName === 'IMG') {
        console.log('新图片出现了:', node.src);
        // 可以在这里做懒加载初始化
      }
    });
  });
});

observer.observe(document.body, {
  childList: true,
  subtree: true
});

六、性能注意事项

MutationObserver虽然比轮询好,但也不能滥用。以下几点要注意:

  1. 不要观察整个document:如果你observe(document.body, { subtree: true, childList: true, attributes: true }),那页面上的任何变化都会触发回调,频繁执行可能影响性能。尽量把观察范围缩小到具体容器。

  2. 回调里不要做太重的操作:MutationObserver的回调是在微任务中执行的,如果里面操作DOM或者计算太多,会阻塞后续渲染。

  3. 及时disconnect:如果不再需要观察,记得调用disconnect()释放资源。

  4. 使用takeRecords():在disconnect之前,可以调用observer.takeRecords()取出尚未处理的变化记录。

七、与旧API对比:Mutation Events的悲惨往事

很久以前,浏览器有一套Mutation Events(比如DOMNodeInsertedDOMAttrModified等)。它们的问题很多:

  • 性能差,每次变化都同步触发,容易导致重入和崩溃
  • 不支持批量观察
  • 被标记为废弃

MutationObserver是它们的完美替代,异步、批量、性能好。

八、总结:MutationObserver就是你的“鹰眼”

  • 它能监听DOM树的各种变化:属性、子节点、文本内容。
  • 配置灵活,可以精确到特定属性或是否包含后代。
  • 异步回调,批量返回变化记录,性能优秀。
  • 应用场景:监听动态内容加载、检测第三方脚本篡改、实现数据绑定(比如某些MVVM库的底层)、与React/Vue的虚拟DOM配合调试等。

有了MutationObserver,你就可以在DOM变化时第一时间响应,像一个隐形的守护者。明天我们将进入Web Storage的世界,看看localStorage、sessionStorage和IndexedDB怎么帮你把数据存到用户浏览器里。

如果你觉得今天的“卧底”够犀利,点个赞让更多人看到。我们明天见!

不用 WebSocket 库,在 React 中构建实时功能

一提到"实时",开发者就会想到 WebSocket 库。Socket.IO、Pusher、Ably -- 生态中有太多选择了。但很多实时功能根本不需要双向通信。股票行情、通知推送、部署日志、实时比分 -- 这些都是服务器到客户端的单向数据流。对于这类场景,浏览器有一个更简单、更轻量、还能自动重连的内置协议:Server-Sent Events(SSE)

将 SSE 与用于连接感知的 Network Information API 和用于跨标签页协调的 BroadcastChannel API 结合起来,你就拥有了一套完整的实时工具包 -- 不需要任何 WebSocket 库。本文将先从零开始手动构建每个部分,看看手动实现在哪里会遇到瓶颈,然后用 ReactUse 的 Hooks 替换,只需几行代码就能处理所有边缘情况。

1. 使用 useEventSource 接入 Server-Sent Events

什么是 Server-Sent Events?

Server-Sent Events(SSE)是一个标准协议,允许服务器通过普通 HTTP 连接向浏览器推送更新。与 WebSocket 不同,SSE 是单向的 -- 服务器发送,客户端接收。浏览器原生的 EventSource API 开箱即用,自动处理连接管理、自动重连和事件解析。

// 一个基本的 SSE 端点(服务端,仅供参考)
// GET /api/notifications
// Content-Type: text/event-stream
//
// data: {"message": "新的部署已启动"}
// id: 1
//
// data: {"message": "部署完成"}
// id: 2

手动实现

让我们在不使用任何库的情况下,在 React 中连接 SSE 端点。

import { useState, useEffect, useRef } from "react";

function useManualEventSource(url: string) {
  const [data, setData] = useState<string | null>(null);
  const [status, setStatus] = useState<
    "CONNECTING" | "CONNECTED" | "DISCONNECTED"
  >("DISCONNECTED");
  const [error, setError] = useState<Event | null>(null);
  const esRef = useRef<EventSource | null>(null);
  const retriesRef = useRef(0);

  useEffect(() => {
    const connect = () => {
      setStatus("CONNECTING");
      const es = new EventSource(url);
      esRef.current = es;

      es.onopen = () => {
        setStatus("CONNECTED");
        setError(null);
        retriesRef.current = 0;
      };

      es.onmessage = (event) => {
        setData(event.data);
      };

      es.onerror = (err) => {
        setError(err);
        setStatus("DISCONNECTED");
        es.close();
        esRef.current = null;

        // 手动重连逻辑
        retriesRef.current += 1;
        if (retriesRef.current < 5) {
          setTimeout(connect, 1000 * retriesRef.current);
        }
      };
    };

    connect();

    return () => {
      esRef.current?.close();
      esRef.current = null;
    };
  }, [url]);

  return { data, status, error };
}

大约 45 行代码,而且已经存在不少问题:

  • 不支持命名事件。 SSE 支持自定义事件类型(如 event: deploy-status),但 onmessage 只能捕获未命名的消息。要支持命名事件,需要对每种事件类型调用 addEventListener,并在卸载时逐一清理。
  • 重连策略过于简陋。 代码最多重试 5 次,使用线性退避,但无法配置重试次数、延迟时间或失败回调。
  • 无法手动关闭/重新打开。 如果用户导航离开又返回,或者你想在标签页隐藏时暂停数据流,还需要更多的状态跟踪。
  • SSR 会崩溃。 EventSource 在服务端不存在。

使用 useEventSource

ReactUse 的 useEventSource Hook 把这些问题全部解决了。

import { useEventSource } from "@reactuses/core";

function DeploymentLog() {
  const { data, status, error, event, lastEventId, close, open } =
    useEventSource("/api/deployments/stream", ["deploy-start", "deploy-end"], {
      autoReconnect: {
        retries: 5,
        delay: 2000,
        onFailed: () => console.error("SSE 连接彻底失败"),
      },
    });

  return (
    <div>
      <div>
        状态:{status}
        {status === "DISCONNECTED" && (
          <button onClick={open}>重新连接</button>
        )}
        {status === "CONNECTED" && (
          <button onClick={close}>断开连接</button>
        )}
      </div>

      {error && <div className="error">连接发生错误</div>}

      <div className="log-entry">
        <span className="event-type">{event}</span>
        <span className="event-id">#{lastEventId}</span>
        <pre>{data}</pre>
      </div>
    </div>
  );
}

看看你免费获得了什么:

  • 命名事件支持。 第二个参数传入事件名数组,Hook 会监听每一个。event 返回值告诉你触发的是哪种事件类型。
  • 可配置的自动重连。 设置重试次数、重试间隔,以及所有重试耗尽时的回调。
  • 手动关闭和重新打开。 调用 close() 断开连接,open() 重新连接 -- 非常适合在后台标签页中暂停数据流。
  • SSR 安全。 Hook 会防范服务端 EventSource 未定义的情况。
  • Last Event ID 追踪。 lastEventId 让你可以从上次断开的位置继续接收(如果服务器支持的话)。

实际示例:实时通知流

import { useEventSource } from "@reactuses/core";
import { useState, useEffect } from "react";

interface Notification {
  id: string;
  title: string;
  body: string;
  severity: "info" | "warning" | "error";
}

function NotificationFeed() {
  const [notifications, setNotifications] = useState<Notification[]>([]);
  const { data, status, event } = useEventSource(
    "/api/notifications/stream",
    ["info", "warning", "error"],
    {
      autoReconnect: {
        retries: -1, // 无限重试
        delay: 3000,
      },
    }
  );

  useEffect(() => {
    if (data) {
      try {
        const notification: Notification = {
          ...JSON.parse(data),
          severity: event as Notification["severity"],
        };
        setNotifications((prev) => [notification, ...prev].slice(0, 50));
      } catch {
        // 数据格式错误,忽略
      }
    }
  }, [data, event]);

  return (
    <div>
      <h2>
        实时通知
        <span className={`status-dot status-${status.toLowerCase()}`} />
      </h2>
      {notifications.map((n) => (
        <div key={n.id} className={`notification notification-${n.severity}`}>
          <strong>{n.title}</strong>
          <p>{n.body}</p>
        </div>
      ))}
    </div>
  );
}

Hook 管理 SSE 的整个生命周期,你的组件只需要关心数据解析和 UI 渲染。

2. 使用 useFetchEventSource 接入需要认证的 SSE 流

原生 EventSource 的局限

原生 EventSource API 有一个重大限制:无法设置自定义请求头。这意味着不能发送 Authorization: Bearer <token>,不能添加自定义 X-Request-ID,也不能发起带 body 的 POST 请求。如果你的 SSE 端点需要认证,EventSource 就不够用了。

常见的变通方案是把 token 放到查询参数中(/api/stream?token=abc),但这会将凭证泄露到服务器日志、浏览器历史记录和 referrer 头中。这是一种安全反模式。

手动实现

要在 SSE 风格的连接中发送自定义请求头,你需要使用 fetch 配合可读流 -- 然后自己处理分块解析、重连和 abort 信号。

import { useState, useEffect, useRef } from "react";

function useManualFetchSSE(url: string, token: string) {
  const [data, setData] = useState<string | null>(null);
  const [status, setStatus] = useState<string>("DISCONNECTED");
  const abortRef = useRef<AbortController | null>(null);

  useEffect(() => {
    const controller = new AbortController();
    abortRef.current = controller;
    setStatus("CONNECTING");

    const connect = async () => {
      try {
        const response = await fetch(url, {
          headers: {
            Authorization: `Bearer ${token}`,
            Accept: "text/event-stream",
          },
          signal: controller.signal,
        });

        if (!response.ok) throw new Error(`HTTP ${response.status}`);
        if (!response.body) throw new Error("No response body");

        setStatus("CONNECTED");
        const reader = response.body.getReader();
        const decoder = new TextDecoder();
        let buffer = "";

        while (true) {
          const { done, value } = await reader.read();
          if (done) break;

          buffer += decoder.decode(value, { stream: true });
          const lines = buffer.split("\n\n");
          buffer = lines.pop() || "";

          for (const chunk of lines) {
            const dataLine = chunk
              .split("\n")
              .find((l) => l.startsWith("data: "));
            if (dataLine) {
              setData(dataLine.slice(6));
            }
          }
        }
      } catch (err) {
        if (!controller.signal.aborted) {
          setStatus("DISCONNECTED");
          // 重连逻辑写在这里...
        }
      }
    };

    connect();
    return () => controller.abort();
  }, [url, token]);

  return { data, status };
}

已经超过 55 行了,而且还不完整。它不处理命名事件、事件 ID、带退避的重连,也不支持 POST 请求。手动解析 SSE 文本协议容易出错。

使用 useFetchEventSource

ReactUse 的 useFetchEventSource Hook 封装了 @microsoft/fetch-event-source 库,提供了 React 友好的 API。它支持自定义请求头、POST 请求体,以及你需要的所有重连逻辑。

import { useFetchEventSource } from "@reactuses/core";

function AuthenticatedStream() {
  const { data, status, event, error, close, open } = useFetchEventSource(
    "/api/private/stream",
    {
      method: "POST",
      headers: {
        Authorization: `Bearer ${getAccessToken()}`,
        "X-Request-ID": crypto.randomUUID(),
      },
      body: JSON.stringify({
        channels: ["deployments", "alerts"],
      }),
      autoReconnect: {
        retries: 10,
        delay: 2000,
        onFailed: () => {
          // Token 可能已过期 -- 重定向到登录页
          window.location.href = "/login";
        },
      },
      onOpen: () => console.log("数据流已连接"),
      onError: (err) => {
        console.error("数据流错误:", err);
        return 5000; // 5 秒后重试
      },
    }
  );

  return (
    <div>
      <div>连接状态:{status}</div>
      {error && <div className="error">{error.message}</div>}
      <pre>{data}</pre>
    </div>
  );
}

两个 Hook 的核心区别:

特性 useEventSource useFetchEventSource
自定义请求头 不支持 支持
POST 请求 不支持 支持
请求体 不支持 支持
底层技术 原生 EventSource fetch API
自动重连 支持 支持
命名事件 支持(通过数组) 支持(通过 event 字段)

当端点是公开的或使用 cookie 认证时,用 useEventSource。当你需要 token 认证、自定义请求头或 POST 请求时,用 useFetchEventSource

实际示例:AI 聊天流式响应

SSE 是流式 AI 响应的标准协议(OpenAI、Anthropic 等都在使用)。以下是如何用认证构建流式聊天 UI。

import { useFetchEventSource } from "@reactuses/core";
import { useState, useEffect, useCallback } from "react";

function AIChatStream() {
  const [messages, setMessages] = useState<
    Array<{ role: string; content: string }>
  >([]);
  const [input, setInput] = useState("");
  const [streamedResponse, setStreamedResponse] = useState("");

  const { data, status, open, close } = useFetchEventSource(
    "/api/chat/completions",
    {
      method: "POST",
      headers: {
        Authorization: `Bearer ${getApiKey()}`,
      },
      body: JSON.stringify({
        messages,
        stream: true,
      }),
      immediate: false, // 不在挂载时连接
      onOpen: () => setStreamedResponse(""),
    }
  );

  // 累积流式传输的 token
  useEffect(() => {
    if (data) {
      try {
        const parsed = JSON.parse(data);
        const token = parsed.choices?.[0]?.delta?.content;
        if (token) {
          setStreamedResponse((prev) => prev + token);
        }
      } catch {
        // 忽略 [DONE] 或格式错误的数据块
      }
    }
  }, [data]);

  const sendMessage = useCallback(() => {
    if (!input.trim()) return;
    setMessages((prev) => [...prev, { role: "user", content: input }]);
    setInput("");
    open(); // 启动 SSE 数据流
  }, [input, open]);

  return (
    <div className="chat">
      {messages.map((msg, i) => (
        <div key={i} className={`message message-${msg.role}`}>
          {msg.content}
        </div>
      ))}
      {streamedResponse && (
        <div className="message message-assistant">{streamedResponse}</div>
      )}
      <div className="input-row">
        <input
          value={input}
          onChange={(e) => setInput(e.target.value)}
          onKeyDown={(e) => e.key === "Enter" && sendMessage()}
          placeholder="输入消息..."
        />
        <button onClick={sendMessage} disabled={status === "CONNECTING"}>
          发送
        </button>
      </div>
    </div>
  );
}

这里 immediate: false 选项至关重要 -- 我们不希望在组件挂载时就打开连接,而是在用户发送消息时显式调用 open()

3. 使用 useNetwork 和 useOnline 检测网络状态

如果用户离线了,实时功能就毫无用处。更糟糕的是,它们会静默失败 -- SSE 连接断开,fetch 请求挂起,UI 显示过时数据,却没有任何提示。好的实时 UI 应该具备网络感知能力。

手动实现

import { useState, useEffect } from "react";

function useManualNetworkStatus() {
  const [isOnline, setIsOnline] = useState(
    typeof navigator !== "undefined" ? navigator.onLine : true
  );
  const [connectionType, setConnectionType] = useState<string | undefined>();

  useEffect(() => {
    const handleOnline = () => setIsOnline(true);
    const handleOffline = () => setIsOnline(false);

    window.addEventListener("online", handleOnline);
    window.addEventListener("offline", handleOffline);

    // Network Information API(并非所有浏览器都支持)
    const conn = (navigator as any).connection;
    if (conn) {
      const handleChange = () => {
        setConnectionType(conn.effectiveType);
      };
      conn.addEventListener("change", handleChange);
      handleChange();

      return () => {
        window.removeEventListener("online", handleOnline);
        window.removeEventListener("offline", handleOffline);
        conn.removeEventListener("change", handleChange);
      };
    }

    return () => {
      window.removeEventListener("online", handleOnline);
      window.removeEventListener("offline", handleOffline);
    };
  }, []);

  return { isOnline, connectionType };
}

大约 35 行代码只获取了两条信息,而且不追踪下行速度、往返时间、数据节省模式或上次状态变化的时间戳。Network Information API 还使用了带厂商前缀的属性(mozConnectionwebkitConnection),这段代码也没有处理。

使用 useNetwork

useNetwork Hook 返回完整的网络信息。

import { useNetwork } from "@reactuses/core";

function NetworkDebugPanel() {
  const {
    online,
    previous,
    since,
    downlink,
    effectiveType,
    rtt,
    saveData,
    type,
  } = useNetwork();

  return (
    <div className="network-panel">
      <div>
        状态:{online ? "在线" : "离线"}
        {previous !== undefined && previous !== online && (
          <span>
            {" "}
            (之前{previous ? "在线" : "离线"},变化于{" "}
            {since?.toLocaleTimeString()})
          </span>
        )}
      </div>
      <div>连接类型:{type ?? "未知"}</div>
      <div>有效类型:{effectiveType ?? "未知"}</div>
      <div>下行速度:{downlink ? `${downlink} Mbps` : "未知"}</div>
      <div>往返时间:{rtt ? `${rtt}ms` : "未知"}</div>
      <div>数据节省:{saveData ? "已启用" : "已关闭"}</div>
    </div>
  );
}

Hook 处理了所有的厂商前缀、事件监听器和 SSR 安全问题。previoussince 字段特别有用 -- 它们让你可以显示"你在 30 秒前离线了",而不仅仅是"离线"。

使用 useOnline

如果你只需要布尔值,useOnline 更加简洁。它是 useNetwork 的轻量封装,只返回 online 值。

import { useOnline } from "@reactuses/core";

function OfflineBanner() {
  const isOnline = useOnline();

  if (isOnline) return null;

  return (
    <div className="offline-banner">
      你当前处于离线状态,实时更新已暂停。
    </div>
  );
}

实际示例:自适应质量推送

useNetwork 返回的网络信息让你可以根据用户的连接质量调整应用行为。

import { useNetwork } from "@reactuses/core";
import { useMemo } from "react";

function useAdaptivePolling(baseInterval: number) {
  const { online, effectiveType, saveData } = useNetwork();

  const interval = useMemo(() => {
    if (!online) return null; // 离线时停止轮询
    if (saveData) return baseInterval * 4; // 尊重数据节省设置
    switch (effectiveType) {
      case "slow-2g":
      case "2g":
        return baseInterval * 3;
      case "3g":
        return baseInterval * 2;
      case "4g":
      default:
        return baseInterval;
    }
  }, [online, effectiveType, saveData, baseInterval]);

  return interval;
}

function LiveScoreboard() {
  const pollingInterval = useAdaptivePolling(5000);
  const { online, effectiveType } = useNetwork();

  return (
    <div>
      {!online && (
        <div className="banner">离线中 -- 显示缓存的比分</div>
      )}
      {effectiveType === "slow-2g" && (
        <div className="banner">慢速连接 -- 更新频率已降低</div>
      )}
      {/* 使用 pollingInterval 的记分牌内容 */}
    </div>
  );
}

在快速 4G 连接上,记分牌每 5 秒更新一次。在慢速 2G 连接上,每 15 秒更新一次。离线时完全停止,显示缓存数据。用户获得的是其连接条件所能支持的最佳体验。

4. 使用 useBroadcastChannel 实现跨标签页通信

实时数据通常需要在浏览器标签页之间共享。如果用户在三个标签页中打开了你的仪表盘,当一条新通知通过 SSE 到达时,三个标签页都应该显示它 -- 但只有一个标签页应该维护 SSE 连接。BroadcastChannel API 让这成为可能。

手动实现

import { useState, useEffect, useRef, useCallback } from "react";

function useManualBroadcastChannel<T>(channelName: string) {
  const [data, setData] = useState<T | undefined>();
  const channelRef = useRef<BroadcastChannel | null>(null);

  useEffect(() => {
    if (typeof BroadcastChannel === "undefined") return;

    const channel = new BroadcastChannel(channelName);
    channelRef.current = channel;

    const handleMessage = (event: MessageEvent<T>) => {
      setData(event.data);
    };

    const handleError = (event: MessageEvent) => {
      console.error("BroadcastChannel 错误:", event);
    };

    channel.addEventListener("message", handleMessage);
    channel.addEventListener("messageerror", handleError);

    return () => {
      channel.removeEventListener("message", handleMessage);
      channel.removeEventListener("messageerror", handleError);
      channel.close();
    };
  }, [channelName]);

  const post = useCallback((message: T) => {
    channelRef.current?.postMessage(message);
  }, []);

  return { data, post };
}

这对简单场景够用了,但它不追踪 BroadcastChannel 是否被支持、频道是否已关闭、错误状态或用于去重的时间戳。

使用 useBroadcastChannel

useBroadcastChannel Hook 提供了完整的、类型安全的封装。

import { useBroadcastChannel } from "@reactuses/core";

interface DashboardMessage {
  type: "NEW_DATA" | "USER_ACTION" | "TAB_CLOSING";
  payload?: unknown;
  sourceTab: string;
}

function DashboardSync() {
  const { data, post, isSupported, isClosed, error } = useBroadcastChannel<
    DashboardMessage,
    DashboardMessage
  >({ name: "dashboard-sync" });

  const broadcast = (type: DashboardMessage["type"], payload?: unknown) => {
    post({
      type,
      payload,
      sourceTab: sessionStorage.getItem("tab-id") || "unknown",
    });
  };

  useEffect(() => {
    if (data?.type === "NEW_DATA") {
      // 用来自另一个标签页的数据更新本地状态
      console.log("收到来自标签页的数据:", data.sourceTab, data.payload);
    }
  }, [data]);

  if (!isSupported) {
    return <div>当前浏览器不支持跨标签页同步。</div>;
  }

  return (
    <div>
      <button onClick={() => broadcast("NEW_DATA", { count: 42 })}>
        与其他标签页共享数据
      </button>
      {error && <div className="error">同步出错</div>}
      {isClosed && <div className="warning">频道已关闭</div>}
    </div>
  );
}

这个 Hook 提供了:

  • isSupported -- 在渲染依赖同步的 UI 前检查 BroadcastChannel 是否可用。
  • isClosed -- 知道频道何时被关闭(由你或浏览器关闭)。
  • error -- 处理消息序列化错误。
  • timeStamp -- 当相同数据被多次接收时进行去重。
  • 类型安全 -- 泛型参数 <D, P> 分别对应接收数据类型和发送数据类型。

5. 综合实战:实时监控仪表盘

让我们将这五个 Hook 组合成一个生产级别的实时仪表盘。这个仪表盘:

  • 通过 SSE 接收实时指标(带认证)
  • 检测网络状态并相应调整行为
  • 在标签页之间共享数据,只让一个标签页维护 SSE 连接
  • 向用户展示连接健康状况
import {
  useFetchEventSource,
  useNetwork,
  useOnline,
  useBroadcastChannel,
  useEventSource,
} from "@reactuses/core";
import { useState, useEffect, useCallback, useRef } from "react";

// --- 类型定义 ---

interface MetricEvent {
  timestamp: number;
  cpu: number;
  memory: number;
  requests: number;
  errors: number;
}

interface TabMessage {
  type: "METRIC_UPDATE" | "CLAIM_LEADER" | "RELEASE_LEADER" | "HEARTBEAT";
  payload?: MetricEvent;
  tabId: string;
}

// --- 领导者选举 Hook ---

function useTabLeader(channelName: string) {
  const tabId = useRef(crypto.randomUUID()).current;
  const [isLeader, setIsLeader] = useState(false);
  const { data, post } = useBroadcastChannel<TabMessage, TabMessage>({
    name: channelName,
  });

  useEffect(() => {
    // 挂载时,短暂延迟后尝试获取领导权
    const timer = setTimeout(() => {
      post({ type: "CLAIM_LEADER", tabId });
      setIsLeader(true);
    }, Math.random() * 200);

    return () => {
      clearTimeout(timer);
      post({ type: "RELEASE_LEADER", tabId });
    };
  }, [post, tabId]);

  useEffect(() => {
    if (data?.type === "CLAIM_LEADER" && data.tabId !== tabId) {
      if (data.tabId > tabId) {
        setIsLeader(false);
      }
    }
    if (data?.type === "RELEASE_LEADER") {
      // 另一个标签页释放了 -- 尝试获取领导权
      setTimeout(() => {
        post({ type: "CLAIM_LEADER", tabId });
        setIsLeader(true);
      }, Math.random() * 100);
    }
  }, [data, tabId, post]);

  return { isLeader, tabId };
}

// --- 网络感知 SSE Hook ---

function useMetricsStream(enabled: boolean) {
  const { online, effectiveType } = useNetwork();

  const { data, status, error, close, open } = useFetchEventSource(
    "/api/metrics/stream",
    {
      headers: {
        Authorization: `Bearer ${getAccessToken()}`,
      },
      immediate: false,
      autoReconnect: {
        retries: -1,
        delay: effectiveType === "4g" ? 2000 : 5000,
        onFailed: () => console.error("指标数据流彻底失败"),
      },
    }
  );

  // 根据 enabled 标志和在线状态连接/断开
  useEffect(() => {
    if (enabled && online) {
      open();
    } else {
      close();
    }
  }, [enabled, online, open, close]);

  return { data, status, error };
}

// --- 主仪表盘组件 ---

function RealtimeDashboard() {
  const [metrics, setMetrics] = useState<MetricEvent[]>([]);
  const isOnline = useOnline();
  const { online, effectiveType, rtt } = useNetwork();

  // 领导者选举 -- 只有领导者标签页打开 SSE 连接
  const { isLeader, tabId } = useTabLeader("metrics-leader");

  // SSE 数据流 -- 只在当前标签页是领导者时激活
  const { data: sseData, status: sseStatus } = useMetricsStream(isLeader);

  // 跨标签页数据共享
  const { data: tabData, post: broadcastToTabs } = useBroadcastChannel<
    TabMessage,
    TabMessage
  >({ name: "metrics-data" });

  // 当领导者收到 SSE 数据时,广播给其他标签页
  useEffect(() => {
    if (isLeader && sseData) {
      try {
        const metric: MetricEvent = JSON.parse(sseData);
        setMetrics((prev) => [...prev, metric].slice(-100));
        broadcastToTabs({
          type: "METRIC_UPDATE",
          payload: metric,
          tabId,
        });
      } catch {
        // 数据格式错误
      }
    }
  }, [isLeader, sseData, broadcastToTabs, tabId]);

  // 当非领导者标签页收到广播数据时,更新本地状态
  useEffect(() => {
    if (!isLeader && tabData?.type === "METRIC_UPDATE" && tabData.payload) {
      setMetrics((prev) => [...prev, tabData.payload!].slice(-100));
    }
  }, [isLeader, tabData]);

  const latestMetric = metrics[metrics.length - 1];

  return (
    <div className="dashboard">
      {/* 连接状态栏 */}
      <header className="status-bar">
        <div className="status-indicators">
          <span className={`dot ${isOnline ? "green" : "red"}`} />
          <span>
            {isOnline ? "在线" : "离线"}
            {effectiveType && ` (${effectiveType})`}
            {rtt && ` -- ${rtt}ms 往返`}
          </span>
        </div>
        <div className="tab-info">
          {isLeader ? "领导者标签页(SSE 活跃)" : "跟随者标签页(通过广播)"}
          <span className={`dot ${sseStatus === "CONNECTED" ? "green" : "yellow"}`} />
        </div>
      </header>

      {/* 离线提示 */}
      {!isOnline && (
        <div className="offline-banner">
          你当前处于离线状态。正在显示最近 {metrics.length} 条缓存指标。
          连接恢复后数据将自动继续更新。
        </div>
      )}

      {/* 指标网格 */}
      {latestMetric && (
        <div className="metrics-grid">
          <MetricCard
            label="CPU 使用率"
            value={`${latestMetric.cpu.toFixed(1)}%`}
            status={latestMetric.cpu > 80 ? "danger" : "normal"}
          />
          <MetricCard
            label="内存"
            value={`${latestMetric.memory.toFixed(1)}%`}
            status={latestMetric.memory > 90 ? "danger" : "normal"}
          />
          <MetricCard
            label="请求数/秒"
            value={latestMetric.requests.toLocaleString()}
            status="normal"
          />
          <MetricCard
            label="错误数/秒"
            value={latestMetric.errors.toLocaleString()}
            status={latestMetric.errors > 10 ? "danger" : "normal"}
          />
        </div>
      )}

      {/* 迷你图表(最近 100 个数据点) */}
      <div className="chart-section">
        <h3>CPU 变化趋势</h3>
        <div className="sparkline">
          {metrics.map((m, i) => (
            <div
              key={i}
              className="bar"
              style={{
                height: `${m.cpu}%`,
                backgroundColor: m.cpu > 80 ? "#ef4444" : "#22c55e",
              }}
            />
          ))}
        </div>
      </div>
    </div>
  );
}

function MetricCard({
  label,
  value,
  status,
}: {
  label: string;
  value: string;
  status: "normal" | "danger";
}) {
  return (
    <div className={`metric-card metric-${status}`}>
      <div className="metric-label">{label}</div>
      <div className="metric-value">{value}</div>
    </div>
  );
}

每个 Hook 在这个仪表盘中的贡献:

  • useFetchEventSource -- 连接带认证的指标 SSE 端点,自动重连。
  • useEventSource -- 如果端点不需要自定义请求头,可以替换使用(对组件零 API 变更)。
  • useNetwork -- 为状态栏提供连接质量数据(effectiveTypertt),并实现自适应重连延迟。
  • useOnline -- 驱动离线提示,在网络断开时暂停 SSE 连接。
  • useBroadcastChannel -- 实现领导者选举和跨标签页数据共享,只让一个标签页维护 SSE 连接,而所有标签页都显示实时数据。

最终效果:

  1. 所有标签页共享一个 SSE 连接(节省服务器资源)
  2. 根据连接质量自适应退避重连
  3. 向用户展示实时网络状态
  4. 离线时优雅降级
  5. 所有打开的标签页之间即时共享数据

选择哪个 Hook

场景 Hook 原因
公开 SSE 端点 useEventSource 简单,原生 EventSource
带认证头的 SSE useFetchEventSource 通过 fetch 支持自定义请求头
带 POST 请求体的 SSE useFetchEventSource 支持请求体
简单的在线/离线检测 useOnline 返回单个布尔值
详细的连接信息 useNetwork 下行速度、往返时间、有效类型
跨标签页消息 useBroadcastChannel 内存通信,无持久化
跨标签页 + 持久化 useBroadcastChannel + useLocalStorage 两全其美

安装

npm install @reactuses/core

或使用你偏好的包管理器:

pnpm add @reactuses/core
yarn add @reactuses/core

相关 Hooks

  • useEventSource -- 响应式 Server-Sent Events,支持命名事件和自动重连
  • useFetchEventSource -- 基于 fetch 的 SSE,支持自定义请求头、POST 请求和认证
  • useNetwork -- 详细的网络状态,包括连接类型、下行速度和往返时间
  • useOnline -- 简单的在线/离线布尔值检测
  • useBroadcastChannel -- 通过 BroadcastChannel API 实现类型安全的跨标签页消息传递
  • useDocumentVisibility -- 跟踪当前标签页是否可见
  • useLocalStorage -- 具有自动跨标签页同步的持久化状态

ReactUse 提供了 100+ 个 React Hooks。探索全部 →

手撕代码之事件委托

一、题目

请补全 JavaScript 代码,要求如下:

  1. ul 标签添加点击事件
  2. 当点击某 li 标签时,该标签内容拼接 . 符号。如:某 li 标签被点击时,该标签内容为 ..

注意: 必须使用 DOM0 级标准事件(onclick

<!DOCTYPE html>
<html>
<head>
    <meta charset="UTF-8">
    <style>
        /* 填写样式 */
    </style>
</head>
<body>
    <ul>
        <li>.</li>
        <li>.</li>
        <li>.</li>
    </ul>
    <!-- 填写标签 -->
    <script type="text/javascript">
        // 填写JavaScript
        document.querySelector('ul').onclick = event => {

        }
    </script>
</body>
</html>

二、思路与笔记

1. 事件委托的核心概念

事件冒泡

提到事件委托,首先会先引入一个 事件冒泡 的概念。

事件冒泡 可以去读一读 MDN 的文档,当然在 React 等前端框架的基础下我认为这个可以暂且不提。

核心思想说,事件冒泡在描述一个浏览器对于嵌套元素的事件是如何处理的。

在子元素上面增加事件处理器时,事件会一层层往父级元素冒泡,父级的父级……

MDN 这里有原话是:

在这种情况下:

  • 最先触发按钮上的单击事件
  • 然后是按钮的父元素(<div> 元素)
  • 然后是 <div> 的父元素(<body> 元素)

我们可以这样描述:事件从被点击的最里面的元素 冒泡 而出。

这种行为可能是有用的,也可能引起意想不到的问题。在接下来的章节中,我们将看到它引起的一个问题,并找到解决方案。

对于一些功能我们并不希望冒泡的,会有 stopPropagation(),当然在今天的手撕代码里并不是重点。

事件委托

事件委托

事件冒泡可以实现 事件委托

在这种做法中,当我们想在用户与大量的子元素中的任何一个互动时运行一些代码时,我们在它们的父元素上设置事件监听器,让发生在它们身上的事件冒泡到它们的父元素上,而不必在每个子元素上单独设置事件监听器。

事件委托就是“自己不处理,让祖先元素代为处理”。

事件委托 = 把监听器挂到父级,利用冒泡 + event.target 来统一处理子元素事件。

事件对象中的 target 属性指向 实际触发事件的元素(不是绑定事件的元素)。

具体来说:

你不必给每个子元素单独绑定事件监听器,而是把监听器绑定在它们的 父元素(或更高层祖先) 上。

当子元素上的事件(比如点击)触发后,事件会沿着 DOM 树 向上冒泡,被父元素捕获并执行对应的处理函数。

示例1

假设有一个 <ul> 列表,里面有 1000 个 <li>

<ul id="list">
    <li>项目 1</li>
    <li>项目 2</li>
    <li>项目 3</li>
    ……
</ul>

不用事件委托的做法:

const items = document.querySelectorAll('#list li');
items.forEach(item => {
    item.addEventListener('click', () => {
        console.log('点击了', item.textContent);
    });
});

缺点:

  • 如果动态新增 <li>,新增的项不会有点击事件(除非重新绑定)
  • 性能差(1000 个监听器)

用事件委托的做法:

const parent = document.getElementById('list');
parent.addEventListener('click', (event) => {
    // event.target 是真正被点击的元素
    if (event.target.tagName === 'LI') {
        console.log('点击了', event.target.textContent);
    }
});

优点:

  • 只有 一个 监听器,性能好
  • 动态新增的子元素 自动具备 点击响应
  • 代码更简洁
示例2

场景:一个待办事项列表,我只关心“项目 3”是否被点击

<ul id="list">
    <li id="item-1">项目 1</li>
    <li id="item-2">项目 2</li>
    <li id="item-3">项目 3</li>
    <li id="item-4">项目 4</li>
</ul>

情况1:直接监听(不是事件委托)

const specificLi = document.getElementById('item-3');
specificLi.addEventListener('click', () => {
    console.log('点击了第3项');
});

特征:

  • 监听器直接挂在 item-3 这个 <li>
  • 不依赖冒泡机制
  • 如果 item-3 被删除再重新添加,需要重新绑定
  • 只监听这一个元素

情况2:事件委托(只关心特定子元素)

const parent = document.getElementById('list');
parent.addEventListener('click', (event) => {
    const li = event.target.closest('li');
    if (li && li.id === 'item-3') {
        console.log('点击了第3项');
    }
});

特征:

  • 监听器挂在父元素 <ul>
  • 依赖冒泡机制:子元素点击 → 事件冒泡到 <ul> → 执行回调
  • 即使 item-3 被删除后重新动态添加,仍然自动有效
  • 一个监听器覆盖了所有子元素,但通过条件过滤只处理 item-3

很多人误以为事件委托只能用来批量处理所有子元素(比如给所有 <li> 加点击)。

实际上,事件委托的核心是利用冒泡 + 祖先监听,至于处理哪些子元素,由条件判断灵活控制

// 可以处理特定子元素
if (li.id === 'item-3') { ... }

// 可以处理某一类子元素
if (li.classList.contains('important')) { ... }

// 可以处理所有子元素(最常用)
if (li) { ... }

Event 接口

Event 接口表示在 EventTarget 上出现的事件。

一些事件是由用户触发的,例如鼠标或键盘事件;或者由 API 生成以表示异步任务的进度。事件也可以通过编程方式触发,例如对元素调用 HTMLElement.click() 方法,或者定义一些自定义事件,再使用 EventTarget.dispatchEvent() 方法将自定义事件派发往指定的目标(target)。

有许多不同类型的事件,其中一些使用基于 Event 主接口的其他接口。Event 本身包含适用于所有事件的属性和方法。

很多 DOM 元素可以被设计接收(或者监听)这些事件,并且执行代码去响应(或者处理)它们。通过 EventTarget.addEventListener() 方法可以将事件处理器绑定到不同的 HTML 元素上(比如 <button><div><span> 等等)。这种方式基本替换了老版本中使用 HTML 事件处理器属性的方式。此外,在正确添加后,还可以使用 removeEventListener() 方法移除这些事件处理器。

备注: 一个元素可以绑定多个事件处理器,甚至是对于完全相同的事件。尤其是相互独立的代码模块出于不同的目的附加事件处理器。(比如,一个网页同时有着广告模块和统计模块同时监听视频播放。)

当有很多嵌套的元素,每个元素都有着自己的事件处理器,事件处理过程会变得非常复杂。尤其当一个父元素和子元素绑定完全相同的事件时,因为结构上的重叠,事件在技术层面发生在两个元素中,触发的顺序取决于每个处理器的事件冒泡的设置。


2. Event.target 属性

Event 接口的 target 只读属性是对事件被分派到的对象的引用。当事件处理器在事件的冒泡或捕获阶段被调用时,它与 Event.currentTarget 不同。

event.target 属性可以用于实现 事件委托

  • event.target实际被点击的那个元素,不是绑定事件的元素。
  • ul.onclick,点击 li 时,event.target 就是那个 li,不是 ul。

在 JS 里,对象传递的就是引用(内存地址的拷贝),你可以直接修改它。

MDN 中有这个示例代码:

// 创建列表
const ul = document.createElement("ul");
document.body.appendChild(ul);

const li1 = document.createElement("li");
const li2 = document.createElement("li");
ul.appendChild(li1);
ul.appendChild(li2);

function hide(evt) {
    // evt.target 指向被点击的 <li> 元素
    // 这与 evt.currentTarget 不同,后者在这个上下文中将指向父级 <ul>
    evt.target.style.visibility = "hidden";
}

// 将监听器附加到列表上
// 点击每个 <li> 时都会触发
ul.addEventListener("click", hide, false);

3. Node.textContent 属性

Node 接口的 textContent 属性表示一个节点及其后代的文本内容。

textContent 的 MDN 文档,它 既可以读,也可以写

innerHTML 的区别

正如其名称,Element.innerHTML 返回 HTML。通常,为了在元素中检索或写入文本,人们使用 innerHTML。但是,textContent 通常具有更好的性能,因为文本不会被解析为 HTML。

此外,使用 textContent 可以防止 XSS 攻击

textContent 获取的是元素内的纯文本内容 string,会过滤掉所有 HTML 标签,只返回文字部分。设置时,内容会作为普通文本插入,不会被解析成 HTML 标签。


4. Element.tagName 属性

注意 tagName 获取当前元素标签名。

if (被点击的元素是 li) {
    获取当前文本内容
    新内容 = 当前内容 + "."
    把新内容设置回去
}

自己乱写了一波

if (event.target.tagName === 'li') {
    let text = event.target.texContent ;
    let newText = text + '.';
    // 不知道怎么设置回去
    event.target.textContent = newText;
}

注意这个不知道怎么设置回去就是依赖 event.target.textContent 是可以读也可以写的。

语法

elementName = element.tagName
  • elementName 是一个字符串,包含了 element 元素的标签名。

备注

在 XML(或者其他基于 XML 的语言,比如 XHTML, xul)文档中,tagName 的值会保留原始的大小写。

比如 span 会返回 SPAN,那在判定的时候要写成大写的。

在 HTML 文档中,tagName 会返回其大写形式。对于元素节点来说,tagName 属性的值和 nodeName 属性的值是相同的。


三、解法

1. 思路推导

根据以上笔记,解题思路如下:

  • 使用 DOM0 级事件 onclick 给 ul 绑定点击事件
  • 通过 event.target 获取实际点击的元素
  • 通过 tagName 判断点击的是否为 li 元素
  • 通过 textContent 读取并修改 li 的内容

2. 最终代码

<!DOCTYPE html>
<html>
<head>
    <meta charset="UTF-8">
    <style>
        /* 填写样式 */
    </style>
</head>
<body>
    <ul>
        <li>.</li>
        <li>.</li>
        <li>.</li>
    </ul>
    <!-- 填写标签 -->
    <script type="text/javascript">
        document.querySelector('ul').onclick = event => {
            let eventName = event.target.tagName;
            if (eventName === 'LI') {
                let eventText = event.target.textContent;
                event.target.textContent = eventText + '.';
            }
        }
    </script>
</body>
</html>

四、小结

还是比较简单的,第一次手撕有点没有习惯这个写法,具体思路很清晰,多查查 API,多看 MDN 的示例。

参考链接

别再手写代码了!2026 前端 5 个 AI 杀招,直接解放 80% 重复劳动(附工具+步骤)

你还在手动搭项目、手写组件、熬夜调 Bug 吗?2026 年的前端开发,AI 已经接管 80% 重复工作——从项目初始化、UI 生成、Bug 修复到代码重构,全流程智能化。

今天这篇,不讲虚的,直接带工具、带步骤、带实战指令,照着做,今天就能少加班 50%。


一、AI 一键搭项目:1 分钟搞定 Vue/React 工程(VS Code + Copilot)

以前搭项目:装依赖、配路由、装状态库、调 ESLint……半天没了。 现在用 GitHub Copilot(VS Code 必装),一句话生成完整工程。

工具安装(5 分钟)

  1. 安装 VS Code(最新版)

image.png

  1. 扩展商店搜:GitHub Copilot + GitHub Copilot Chat(安装)

image.png

  1. 点击左下角 Copilot 图标 → 登录 GitHub → 授权成功(图标变绿)

image.png

实战步骤(1 分钟出项目)

  1. 新建空文件夹 → 用 VS Code 打开
  2. 快捷键 Ctrl+Shift+I(Win)/ Cmd+Shift+I(Mac)打开 Copilot Chat

image.png

  1. 直接发指令(复制可用):

    生成一个 Vue3 + Vite + Pinia + VueRouter + Tailwind CSS 项目,包含:

    • 完整目录结构
    • ESLint + Prettier 规范配置
    • 请求封装(axios)
    • 路由守卫
    • 自适应布局基础
    • 自动安装依赖
  2. 等待 30 秒 → AI 自动生成所有文件、安装依赖、写好 README

image.png

效果对比

  • 以前:1 天工作量
  • 现在:1 分钟,零配置、零报错

二、AI 组件工厂:一句话生成生产级 UI(Cursor 编辑器)

前端最耗时:写页面、调样式、做响应式、加交互。 Cursor(AI 原生编辑器) 是前端 UI 生成神器,比 VS Code 更智能,支持跨文件、自动处理样式依赖。

工具安装

  1. 官网下载:www.cursor.so/

image.png

  1. 安装 → 首次启动用 GitHub 登录 → 导入 VS Code 配置

安装

  1. 设置中文:Ctrl+Shift+P → 搜索 Configure Display Language → 选中文

设置中文

实战步骤(生成电商商品卡片)

  1. 新建 ProductCard.vue
  2. 快捷键 Ctrl+K(Win)/ Cmd+K(Mac)打开 AI 指令

打开 AI

  1. 输入(复制可用):

    用 Vue3 + TS + Tailwind CSS 生成电商商品卡片组件,要求:

    • 包含:商品图、标题、原价、现价、折扣标签、加入购物车按钮
    • hover 上浮动效、过渡动画
    • 移动端响应式(375px 适配)
    • 带 TS 类型定义
    • 支持自定义主题色
    • 加注释、符合 ESLint 规范
  2. 回车 → 直接生成完整代码(复制即用)

完整代码

进阶:Figma 转代码

  1. 打开 Figma 设计稿 → 复制链接
  2. Cursor 指令:

    把这个 Figma 设计稿转成 Vue3 代码:[粘贴链接],带响应式、TS 类型、可直接运行


三、AI 自动改 Bug:秒定位+修复,告别熬夜(Copilot Chat)

前端最痛:白屏、样式错乱、报错、兼容问题。 Copilot Chat 能直接读代码+报错,自动定位根因+给修复方案

实战步骤(修复白屏 Bug)

  1. 遇到报错:Uncaught TypeError: Cannot read properties of undefined (reading 'xxx')
  2. 选中报错代码 → 右键 → Copilot → Explain This Error

修改Bug

  1. 或直接在聊天框发:

    分析这段代码和报错,找出根因,给修复代码+解释: 【粘贴报错】 【粘贴代码】

  2. AI 秒回:

  • 错误原因(如:变量未初始化、异步时序问题)
  • 完整修复代码
  • 优化建议(如:加可选链、错误捕获)

image

常见前端 Bug 指令(直接复制)

  • 样式兼容:修复 iOS 微信浏览器样式错乱问题
  • 性能卡顿:分析页面滚动卡顿,优化 FPS,给代码方案
  • 接口报错:修复 axios 跨域+超时+错误重试

四、AI 代码重构:老项目一键升级(文心快码)

维护 jQuery/老 Vue2 项目?手动重构太痛苦。 文心快码(国产 AI,前端重构最强) 能批量升级、补 TS、优化性能。

工具安装(VS Code 插件)

  1. 扩展商店搜:文心快码(Baidu Comate) → 安装

Baidu Comate

  1. 用百度账号登录 → 免费额度够用

实战步骤(jQuery 转 Vue3)

  1. 打开老代码文件

  2. 打开文心快码聊天 → 发指令:

    把这段 jQuery 代码重构成 Vue3 组合式 API + TS,要求:

    • 保留原功能
    • 加类型定义
    • 用 Pinia 管理状态
    • 优化性能、移除冗余
    • 符合团队规范
  3. AI 自动生成新代码 → 对比确认 → 直接替换

进阶:批量重构

分析整个项目,把所有 Vue2 组件升级到 Vue3,统一 TS 规范


五、AI 全链路工程化:从接口到部署一条龙(v0 + Copilot)

不止写代码,接口、类型、测试、部署 AI 全包。 v0(Vercel 出品)+ Copilot 前端全链路最强组合。

1. 接口 + TS 类型自动生成

Copilot 指令:

根据这份接口文档,生成:

  • axios 请求封装
  • TS 接口类型定义
  • Mock 数据
  • API 调用示例

2. UI 生成(v0 最强)

  1. 打开:v0.dev/

image

  1. 输入:生成一个后台管理系统列表页,带筛选、分页、操作按钮,用 React + Tailwind
  2. 10 秒出页面 → 复制代码到项目

3. 自动写测试 + 部署

Copilot 指令:

为这个组件写 Vitest 单元测试,覆盖:渲染、交互、边界情况 再生成 Dockerfile + CI/CD 部署脚本


2026 前端 AI 工具选型表(直接抄)

场景 最佳工具 价格 上手难度
日常编码、补全 GitHub Copilot $19/月
UI 组件、页面生成 Cursor、v0 $20/月(Cursor) ⭐⭐
老项目重构、升级 文心快码 免费额度+付费
Bug 修复、调试 Copilot Chat 含在 Copilot 内
全栈项目、原型 Bolt.new 免费试用 ⭐⭐

最后:AI 不淘汰前端,淘汰不用 AI 的人

2026 年的前端竞争:

  • ❶ 不会 AI:天天手写、加班、被淘汰
  • ❷ 会用 AI:少写 80% 重复代码、早下班、涨薪更快

今天就行动

  1. 装 VS Code + Copilot + Cursor
  2. 把本文指令复制试用
  3. 把重复工作丢给 AI,专注架构、业务、价值

别再手写代码了,AI 时代,拼的是会不会用工具,不是手速!


各位互联网搭子,要是这篇文章成功引起了你的注意,别犹豫,关注、点赞、评论、分享走一波,让我们把这份默契延续下去,一起在知识的海洋里乘风破浪!

前端工程师必备的 10 个 AI 万能提示词(Prompt),复制直接用,效率再翻倍!

你是不是也有这种困扰? 用 Copilot、Cursor 写代码,明明想让 AI 帮你省时间,结果指令发出去,AI 瞎编代码、答非所问,反而更费劲儿?

不是 AI 不好用,是你没找对“说话方式”——前端 AI 高效开发的核心,从来不是“让 AI 写代码”,而是“让 AI 精准懂你的需求”。

很多前端每天用 AI,却不知道:一句好的 Prompt(提示词),能让效率直接翻 3 倍,少写 80% 重复代码、少踩 90% 的坑。

今天这篇,不搞虚的,直接给大家整理了 10 个前端专属 AI 万能提示词,覆盖前端开发全场景——组件开发、Bug 修复、代码重构、样式优化、工程化配置,全部复制就能用,不用自己琢磨,新手也能轻松上手。

不管你用的是 Copilot、Cursor、文心快码,还是 Claude Code,这些 Prompt 都通用,今天用,今天就能省时间、少加班!

先划重点:前端 AI Prompt 万能公式(记牢更省心)

所有好用的前端 Prompt,都离不开这 4 个核心要素,记下来,以后自己也能自定义:

明确场景 + 技术栈 + 具体需求 + 输出要求

举个反例:“帮我写个按钮组件”(模糊,AI 易瞎编) 举个正例:“用 Vue3 + TS + Tailwind CSS 写一个按钮组件,包含默认/禁用/高亮三种状态,hover 有过渡动画,带类型定义和注释,符合 ESLint 规范”(精准,AI 直接出可用代码)

下面这 10 个 Prompt,全部按照这个公式编写,复制粘贴,替换括号里的内容,就能直接用!

一、组件开发类(最常用,每天都能用到)

前端每天都要写组件,这 2 个 Prompt,覆盖 80% 的组件开发场景,不用再手动写样式、写逻辑。

Prompt 1:基础组件生成(复制即用)

用【Vue3/React】+【TS】+【Tailwind CSS/Element Plus/Ant Design】生成【组件名称,如:登录表单/商品卡片/分页组件】,要求:
1. 包含【具体功能,如:表单校验/分页切换/hover 动效】;
2. 支持【自定义属性,如:自定义颜色/尺寸/回调函数】;
3. 带完整 TS 类型定义、详细注释,符合 ESLint 规范;
4. 适配移动端响应式,兼容主流浏览器;
5. 输出完整可运行代码,复制就能直接导入项目。

示例替换:用 Vue3 + TS + Element Plus 生成登录表单,要求:1. 包含账号密码校验、记住密码、忘记密码功能;2. 支持自定义提交按钮文本;3. 带完整 TS 类型定义、详细注释,符合 ESLint 规范;4. 适配移动端响应式,兼容主流浏览器;5. 输出完整可运行代码,复制就能直接导入项目。

Prompt 2:复杂组件封装(复制即用)

帮我封装一个【复杂组件名称,如:树形表格/弹窗表单/下拉搜索选择器】,技术栈【Vue3/React + TS】,要求:
1. 核心功能:【详细描述功能,如:树形表格支持勾选、展开/折叠、搜索筛选;弹窗表单支持表单联动、提交校验】;
2. 性能优化:【如:懒加载、防抖节流、避免重复渲染】;
3. 可扩展性:支持插槽、自定义事件、Props 传参,方便后续二次开发;
4. 附带使用示例、TS 类型说明、常见问题备注;
5. 代码结构清晰,分模块编写,便于维护。

二、Bug 修复类(前端救星,告别熬夜改 Bug)

遇到 Bug 不用慌,不用再翻 Stack Overflow、不用瞎试代码,这 2 个 Prompt,让 AI 秒定位、秒修复,还能告诉你问题根源。

Prompt 3:报错快速修复(复制即用)

帮我分析以下前端报错和对应代码,要求:
1. 报错信息:【粘贴完整报错信息,如:Uncaught TypeError: Cannot read properties of undefined (reading 'value')】;
2. 对应代码:【粘贴报错相关的完整代码片段】;
3. 请找出报错根因,给出详细解释,然后提供完整的修复代码;
4. 补充优化建议,避免以后再出现类似问题;
5. 修复后的代码要符合项目技术栈【Vue3/React + TS】规范,可直接替换使用。

Prompt 4:兼容性/Bug 排查(复制即用)

我遇到一个前端问题:【详细描述问题,如:iOS 微信浏览器样式错乱、页面滚动卡顿、接口请求跨域失败、组件渲染异常】;
项目技术栈:【Vue3/React + TS + 具体框架/工具】;
请帮我:
1. 分析可能的问题原因,列出所有可能性;
2. 给出每一种原因的解决方案和完整代码;
3. 提供预防措施,避免后续出现类似兼容性/性能问题;
4. 方案要简单易操作,不用复杂配置,直接能落地。

三、代码重构类(老项目救星,提升代码质量)

维护老项目、接手烂代码,手动重构太费时间?这 2 个 Prompt,让 AI 帮你优化代码、升级版本,不用自己逐行修改。

Prompt 5:代码优化/重构(复制即用)

帮我重构以下前端代码,项目技术栈【Vue3/React + TS】,要求:
1. 原始代码:【粘贴需要重构的代码片段】;
2. 重构目标:优化代码结构、移除冗余代码、修复潜在 Bug、提升代码可读性和可维护性;
3. 保留原有的所有功能,不改变业务逻辑;
4. 加入 TS 类型定义(如果没有),补充必要注释,符合 ESLint 规范;
5. 给出重构前后的对比说明,解释优化的原因和好处。

Prompt 6:版本升级迁移(复制即用)

帮我将【旧版本技术,如:Vue2 组件/Vue3 旧语法/jQuery 代码】迁移到【新版本技术,如:Vue3 组合式 API/TS/React 函数组件】,要求:
1. 原始代码:【粘贴需要迁移的代码片段/文件】;
2. 迁移要求:完全保留原业务功能,兼容原有项目配置,不引入新的依赖;
3. 遵循新版本的最佳实践,如:Vue3 组合式 API 规范、React Hooks 规范;
4. 补充迁移说明,列出需要注意的细节和可能出现的问题及解决方案;
5. 输出完整的迁移后代码,可直接替换使用。

四、样式/交互类(告别调样式的痛苦)

调样式、做交互,最费时间还容易出错?这 2 个 Prompt,让 AI 帮你写样式、做动效,不用再反复调试。

Prompt 7:样式快速生成/优化(复制即用)

帮我写/优化【元素/组件】的样式,技术栈【Tailwind CSS/CSS3/SCSS】,要求:
1. 样式需求:【详细描述,如:居中显示、圆角、阴影、hover 动效、响应式适配(375px/768px/1200px)、深色模式兼容】;
2. 样式规范:符合项目设计规范,避免样式冲突,代码简洁可复用;
3. 优化要求:减少冗余样式,提升样式加载速度,兼容主流浏览器;
4. 输出完整的样式代码,可直接复制到项目中使用,并给出使用说明。

Prompt 8:交互效果实现(复制即用)

帮我实现【交互效果,如:下拉菜单动画、弹窗淡入淡出、滚动加载、拖拽排序、表单联动】,技术栈【Vue3/React + JS/TS】,要求:
1. 交互细节:【详细描述,如:弹窗点击遮罩关闭、下拉菜单hover展开、拖拽时显示提示、滚动加载到底部自动请求数据】;
2. 性能要求:避免卡顿、防抖节流处理,不影响页面其他功能;
3. 兼容性:适配移动端和PC端,兼容主流浏览器;
4. 输出完整的代码(HTML/CSS/JS/TS),复制就能用,附带使用说明和注意事项。

五、工程化/工具类(提升全链路效率)

除了写代码,工程化配置、接口请求、测试用例也能让 AI 帮你做,这 2 个 Prompt,覆盖前端全链路开发。

Prompt 9:接口请求/类型生成(复制即用)

根据以下接口文档,生成【Vue3/React】项目的接口请求代码,要求:
1. 接口信息:【粘贴接口文档,包含请求地址、请求方式、参数、返回值】;
2. 技术栈:【Axios + TS】;
3. 输出内容:
   - 完整的接口请求函数封装(包含请求拦截、响应拦截、错误处理);
   - 所有接口参数和返回值的 TS 类型定义;
   - Mock 数据生成(用于本地调试);
   - 接口调用示例;
4. 代码符合项目规范,可直接导入项目使用。

Prompt 10:测试用例/工程化配置(复制即用)

帮我生成【组件/函数】的测试用例,或【工程化配置文件】,要求:
1. 目标:【如:为登录组件写单元测试、生成 ESLint 配置、生成 Vitest 配置、生成 Dockerfile】;
2. 技术栈:【Vitest/Jest/ESLint/Docker】;
3. 具体要求:【如:测试用例覆盖渲染、交互、边界情况;配置文件适配 Vue3/React + TS 项目,包含常用配置】;
4. 输出完整的代码/配置文件,可直接复制到项目中使用,并给出配置说明和使用方法。

关键提醒:这 3 个小技巧,让 Prompt 效果再翻倍

  1. 越具体,AI 越精准:不要说“帮我写个表单”,要明确技术栈、功能、样式,甚至是兼容要求,避免 AI 瞎编;
  2. 分场景使用:不同的 AI 工具(Copilot/Cursor)适配性略有差异,但以上 Prompt 全部通用,复制后可根据工具微调;
  3. 善用追问:如果 AI 输出不符合预期,直接追问“修改一下,让组件支持自定义颜色”“修复这个代码里的语法错误”,不用重新发指令。

写在最后:AI 提效,Prompt 是关键

2026 年的前端开发,拼的不是手速,是“用 AI 的能力”——同样是用 AI,会写 Prompt 的人,每天能多省 1-2 小时,少加很多班;不会写的人,反而被 AI 拖累。

以上 10 个 Prompt,覆盖了前端开发的全场景,不管你是新手还是资深前端,复制就能用,不用自己琢磨、不用记复杂语法。

建议你 收藏本文,转发给身边还在瞎用 AI、天天加班的前端同事,一起省时间、提效率、早下班。


各位互联网搭子,要是这篇文章成功引起了你的注意,别犹豫,关注、点赞、评论、分享走一波,让我们把这份默契延续下去,一起在知识的海洋里乘风破浪!

前端进阶之路:从性能优化到响应式布局的实战指南(Tailwindcss)

前端进阶之路:从性能优化到响应式布局的实战指南(Tailwindcss)

在现代前端开发的宏大叙事中,我们往往容易迷失在纷繁复杂的框架和库中。然而,剥开React、Vue或Tailwind CSS的外衣,其核心往往回归到对DOM操作的深刻理解、对性能的极致追求以及对用户体验的细腻把控。今天,我们将串联起三个看似独立却内在逻辑紧密相连的知识点:原生DOM操作的性能基石——DocumentFragment,React组件化开发的“隐形斗篷”——Fragment,以及现代CSS布局的利器——Tailwind CSS的响应式哲学。

一、性能的基石:理解DocumentFragment

在JavaScript直接操作DOM的时代,性能优化是一个绕不开的话题。浏览器渲染页面的过程是昂贵的,每一次DOM的增删改查都可能触发重排和重绘。让我们通过两段代码的对比,来窥探原生性能优化的奥秘。

假设我们需要向页面中的一个容器内插入两个段落元素。

直接追加的“笨重”方式:

const container = document.querySelector('.container');
const p1 = document.createElement('p');
p1.textContent = '111';
const p2 = document.createElement('p');
p2.textContent = '222';

container.appendChild(p1); // 触发一次重排/重绘
container.appendChild(p2); // 再次触发重排/重绘

这种方式虽然直观,但效率极低。每执行一次appendChild,浏览器就需要重新计算样式、布局,并更新画面。如果你需要插入成百上千个节点,页面就会出现明显的卡顿甚至闪烁。

使用DocumentFragment的“聪明”方式:

const container = document.querySelector('.container');
const p1 = document.createElement('p');
p1.textContent = '111';
const p2 = document.createElement('p');
p2.textContent = '222';

const fragment = document.createDocumentFragment(); // 创建内存中的碎片
fragment.appendChild(p1); // 内存操作,无渲染消耗
fragment.appendChild(p2); // 内存操作,无渲染消耗
container.appendChild(fragment); // 一次性插入,仅触发一次重排/重绘

DocumentFragment就像一个存在于内存中的“隐形容器”。它不属于DOM树,因此对它的操作不会触发页面的渲染更新。我们可以把所有的子节点先组装到这个碎片中,最后一次性将其内容“倾倒”进真实的DOM树。这就像搬家时,与其每拿一个箱子就跑一趟新车,不如先把所有箱子装上一辆大卡车(Fragment),然后一次性运达目的地。这种批量操作的思维,是现代前端性能优化的原点。

二、React的“隐形斗篷”:Fragment组件

随着React等声明式框架的普及,我们不再直接操作DOM,但DocumentFragment的思想在React中以另一种形式得到了升华——那就是Fragment组件。

在React中,组件的render函数或函数组件本身必须返回一个单一的根元素。这是由React内部虚拟DOM树的协调机制决定的。

痛点场景:

//  错误写法:返回了多个根节点,React会报错
function MyComponent() {
  return (
    <h1>标题</h1>
    <p>内容</p>
  );
}

为了解决这个问题,新手往往会用一个无意义的div包裹起来:

// ️ 不理想的写法:引入了多余的DOM节点
function MyComponent() {
  return (
    <div>
      <h1>标题</h1>
      <p>内容</p>
    </div>
  );
}

这种做法虽然能跑通,但会带来“DOM污染”。多余的div会破坏CSS布局(比如Flexbox的父子关系),增加DOM树的深度,甚至导致HTML结构语义错误(例如在table的tr中插入div)。

Fragment的解决方案:

React提供了Fragment(简写为<>...</>),它就像一个“隐形斗篷”。它满足了React对单一根节点的要求,但在最终生成的HTML中,它自身会消失,只留下它的子元素。

//  完美写法:使用Fragment,DOM结构纯净
function MyComponent() {
  return (
    <>
      <h1>标题</h1>
      <p>内容</p>
    </>
  );
}

在列表渲染中,Fragment更是不可或缺。它允许我们将一组相关的元素(如术语dt和描述dd)组合在一起,而不破坏父级列表的结构。

// 在列表中组合多个元素,保持语义化
{items.map(item => (
  <Fragment key={item.id}>
    <dt>{item.term}</dt>
    <dd>{item.description}</dd>
  </Fragment>
))}

虽然React的Fragment和原生的DocumentFragment在实现机制上有所不同(前者是虚拟DOM层面的概念,后者是真实DOM API),但它们的精神内核是一致的:高效组织节点,避免冗余,拒绝不必要的渲染开销。

三、布局的艺术:Tailwind CSS与移动端优先

当我们构建好纯净的DOM结构后,如何高效地为其赋予样式?Tailwind CSS提供了一种原子化的解决方案,而“移动端优先”则是其响应式布局的核心哲学。

让我们看一段典型的Tailwind代码:

export default function App() { 
    return (
        <div className="flex flex-col md:flex-row gap-4">
            <main className="bg-blue-100 p-4 md:w-2/3">
                主内容
            </main>
            <aside className="bg-green-100 p-4 md:w-1/3">侧边栏</aside>
        </div>
    )
}

这段代码精妙地展示了如何适配不同设备。

第一阶段:移动端(默认样式) 当我们在手机上(屏幕宽度小于768px)浏览时,Tailwind会忽略所有带前缀(如md:)的类名。

  • flex flex-col:容器启用Flex布局,且方向为垂直堆叠。
  • gap-4:元素间保持间距。 此时,主内容和侧边栏是上下排列的单列布局,非常适合手机阅读。

第二阶段:PC端(md:断点生效) 当屏幕变宽(≥768px),md:前缀的样式被激活。

  • md:flex-row:布局方向瞬间切换为水平排列。
  • md:w-2/3md:w-1/3:主内容占据2/3宽度,侧边栏占据1/3。 页面平滑地过渡为经典的两栏布局。

这种“先写死移动端,再修饰大屏”的策略,避免了复杂的媒体查询嵌套,让响应式逻辑变得清晰可见。它要求开发者首先关注核心内容的呈现(移动端),然后再考虑在大屏幕上如何利用富余空间(PC端),这是一种非常健康的设计思维。

结语

从原生JavaScript中利用DocumentFragment减少重排,到React中使用Fragment保持DOM树的语义化和纯净,再到利用Tailwind CSS的原子类快速构建响应式布局,这三者共同构成了一个现代前端开发者的核心素养。

它们分别解决了结构、逻辑和表现层面的关键问题:

  • DocumentFragment教会我们敬畏浏览器的渲染性能
  • React Fragment教会我们追求代码结构的逻辑纯净
  • Tailwind CSS教会我们以高效的方式驾驭复杂的UI设计

掌握这些工具背后的原理,而不仅仅是语法,才是通往高级前端工程师的必经之路。

前端面试必问 Git 通关指南:常用命令速查 + merge/rebase 深度辨析,看完再也不慌

本文面向前端面试场景,覆盖从日常开发高频命令,到面试官必问的核心原理辨析,所有内容均来自面试实战考点,无冗余废话,面试前刷一遍直接通关。

开篇前言

Git 是前端开发的必备工具,也是几乎所有公司一面必问的基础考点。但很多同学日常开发只会用add/commit/push/pull,一被问到mergerebase的区别、代码回滚方案、分支管理规范就卡壳。

本文基于我备战前端实习的面试笔记整理,先补全优化前端开发必背的全量高频命令,再深度拆解面试最高频的merge vs rebase考点,最后补充面试常问的附加题,帮大家彻底搞定 Git 面试。


一、前端开发 & 面试必背 Git 常用命令大全

我将所有命令按开发场景做了模块拆分,保留了基础核心用法,同时补全了面试常考的进阶参数和注意事项,可直接当作面试速查手册。

1. 仓库初始化与远程关联

表格

命令 核心作用 面试注意事项
git init 在当前目录初始化一个全新的 Git 本地仓库 初始化后会生成.git隐藏目录,存储 Git 所有的版本和配置信息
git clone <远程仓库地址> 克隆远程仓库到本地,自动完成远程关联 面试常考:支持 HTTPS 和 SSH 两种地址,SSH 需提前配置密钥
git remote add origin <远程仓库地址> 给本地仓库关联远程仓库,origin 是远程仓库的默认别名 必用场景:本地 init 的仓库首次推送到远程前,必须先执行此命令
git remote -v 查看当前仓库关联的远程仓库地址详情 排查远程仓库配置问题的核心命令

2. 分支管理核心命令(面试超高频)

表格

命令 核心作用 面试注意事项
git branch 查看所有本地分支,带*的是当前所在分支 基础必背,面试官常以此为起点延伸分支相关问题
git branch -r 查看所有远程分支
git branch -a 查看所有分支(本地 + 远程)
git checkout <分支名> 切换到已存在的本地分支 高频快捷操作:git checkout - 一键切换到上一个分支
git checkout -b <新分支名> 创建并立即切换到新分支 等价于git branch <新分支名> + git checkout <新分支名>,开发最高频用法
git checkout -b <新分支名> <起点> 基于指定起点(远程分支 / 历史提交 / 标签)创建新分支 示例:git checkout -b hotfix/v1.0 origin/main,面试常考场景化用法
git branch -d <分支名> 安全删除本地分支 仅能删除已经合并到当前分支的分支,未合并的分支会报错,防止误删代码
git branch -D <分支名> 强制删除本地分支 无论分支是否合并,都会直接删除,仅用于废弃的功能分支,面试常问-d-D的区别
git push origin --delete <远程分支名> 删除远程分支

补充:Git 2.23+ 官方推出语义更清晰的git switch替代checkout的分支操作功能,面试可提:

  • 切换分支:git switch <分支名>
  • 创建并切换新分支:git switch -c <新分支名>解决了checkout功能过载、容易误操作的问题。

3. 代码提交与暂存核心命令

表格

命令 核心作用 面试注意事项
git status 查看当前工作目录和暂存区的状态,显示未跟踪、已修改、已暂存的文件 开发必用,切换分支、提交代码前建议必执行,避免误操作
git add <文件路径/文件名> 将指定文件的修改 / 新增添加到暂存区
git add . 将当前目录所有修改、新增的文件添加到暂存区 面试常考:不会处理已删除的文件,仅新增和修改
git add -u 仅将已跟踪文件的修改、删除添加到暂存区,不包含新增文件 高频场景:只想提交已有文件的改动,不想提交新增的临时文件
git commit -m "提交描述信息" 将暂存区的内容提交到本地仓库,生成一条提交记录 核心要求:提交信息必须语义化,面试常问提交规范
git commit -am "提交描述信息" 等价于git add -u + git commit -m,一步完成已跟踪文件的提交 注意:对新增的未跟踪文件无效
git commit --amend 修改上一次的提交信息,或补充漏提交的文件,不会生成新的提交 面试高频考点:仅适用于未推送到远程的本地提交,已推送的提交修改后会重写历史,需要强制推送,有极高风险

4. 代码拉取与推送核心命令

表格

命令 核心作用 面试注意事项
git push 将本地当前分支的提交推送到已关联的远程分支 首次推送必须加-u参数设置上游分支:git push -u origin <分支名>,后续可直接用git push
git push -f 强制推送本地分支覆盖远程分支 高危操作!仅能在自己的私有分支使用,绝对禁止在公共分支执行,会直接覆盖远程历史,导致团队代码丢失
git pull 拉取远程当前分支的最新代码,并自动合并到本地分支 面试核心考点:git pull = git fetch + git merge,默认用 merge 方式合并,会生成合并提交
git pull --rebase 拉取远程最新代码,并用 rebase 方式合并到本地分支 多人协作高频用法,避免生成多余的合并提交,保持本地历史线性
git fetch 仅拉取远程仓库的所有最新提交到本地,不会自动合并 面试常问和git pull的区别:更安全,可先查看远程改动,再手动决定是否合并,不会直接修改本地工作区

5. 临时存储 stash 全命令(面试高频)

表格

命令 核心作用 面试注意事项
git stash 将当前分支未提交的改动(工作区 + 暂存区)临时保存到堆栈中,清空工作区 高频场景:正在开发功能,突然需要切换分支改 bug,又不想提交半成品代码
git stash save "存储备注信息" 给 stash 记录添加备注,方便后续识别 多个 stash 记录时必用,避免分不清存储的内容
git stash list 查看堆栈中所有的 stash 存储记录
git stash pop 恢复堆栈中最新的一条 stash 记录,并删除该条记录 恢复后会自动从堆栈中移除,对应git stash的反向操作
git stash apply 恢复最新的 stash 记录,但不会从堆栈中删除 场景:需要把同一份改动恢复到多个分支
git stash drop 删除堆栈中最新的一条 stash 记录
git stash clear 清空堆栈中所有的 stash 记录

6. 提交历史查看命令

表格

命令 核心作用 面试注意事项
git log 查看当前分支的完整提交日志记录,包含提交哈希、作者、时间、提交信息
git log --oneline 一行显示一条提交记录,仅展示简短提交哈希和提交信息 高频用法,快速查看提交历史,面试必提
git log --graph 图形化展示分支的合并历史和分叉情况 配合--oneline使用效果极佳,直观看到分支合并轨迹
git log -p 查看提交日志的同时,显示每次提交的具体代码改动内容 排查问题、代码溯源高频用法

7. 工作区修改撤销与文件恢复

表格

命令 核心作用 面试注意事项
git checkout -- <文件路径/文件名> 用暂存区的版本覆盖工作区的文件,撤销未暂存的修改 ⚠️ 高危操作:修改不可逆,本地未暂存的改动会永久丢失
git checkout -- . 撤销当前目录所有未暂存的修改
git checkout <提交哈希/分支名> -- <文件路径> 用指定提交 / 分支的文件版本,覆盖当前工作区和暂存区的对应文件 场景:恢复某个文件到历史版本,不影响其他文件

补充:Git 2.23+ 官方推出git restore替代checkout的文件恢复功能,语义更清晰:

  • 撤销工作区未暂存修改:git restore <文件名>
  • 撤销暂存区的修改:git restore --staged <文件名>

二、前端面试最高频考点:git merge vs git rebase 深度辨析

这是 Git 面试的必问题,90% 的面试官都会问到,很多同学只能答出 “一个会生成合并提交,一个不会”,但想要拿到高分,必须从原理、区别、优缺点、场景、禁忌全维度讲透。

1. 核心相同点

git mergegit rebase核心目标完全一致:将一个分支的代码变更,整合到另一个分支中,是 Git 中最核心的两种分支合并方案。

2. 核心原理(面试答题先讲原理,直接拉开差距)

我们用一个最常见的开发场景举例:

主分支main有提交记录 A→B→C,我们从C切出功能分支feature开发,提交了D→E;此时main分支有了新的提交F→G,现在需要把main的最新代码合并到feature,或者把feature合并到main

git merge 原理

git merge采用三方合并策略,执行git merge feature时会做 3 件事:

  1. 找到两个分支的最近共同祖先 C
  2. 基于共同祖先,将两个分支的变更做三方合并对比;
  3. 最终生成一个全新的合并提交 H,这个提交有两个父提交,分别指向两个分支的最新提交GE,同时完整保留两个分支的所有原始提交历史。

最终合并后的main分支历史:A→B→C→F→G→H(合并提交)feature分支的D、E提交完整保留,时间线是分叉的。

git rebase 原理

rebase直译是变基,核心是改变分支的基准,执行git rebase main时会做 4 件事:

  1. 找到两个分支的最近共同祖先 C
  2. 提取feature分支上从C之后的所有提交D、E,临时保存起来;
  3. feature分支的基准指针,直接指向main分支的最新提交G
  4. 按顺序将临时保存的D、E,逐个重放应用到新的基准G上,生成新的提交D'、E'

最终变基后的feature分支历史:A→B→C→F→G→D'→E',形成了完全线性的提交记录,没有任何合并提交,原始的D、E提交会被废弃,提交历史被重写。

3. 全维度对比表(面试分点答,逻辑拉满)

表格

对比维度 git merge git rebase
核心逻辑 三方合并,生成全新的合并提交 变基重放,逐个应用提交,重写提交历史
提交历史 完整保留所有分支的原始提交,时间线分叉,上下文完整 重写提交历史,形成线性记录,无多余合并提交,原始上下文丢失
冲突处理 合并时仅需解决 1 次冲突,解决后生成合并提交即可,成本极低 变基过程中,每个提交重放时都可能产生冲突,需要逐个解决,提交越多成本越高
操作安全性 极高,不会修改现有提交历史,所有操作都有记录可追溯,不会丢失代码 高危,会重写提交历史,操作失误极易丢失提交,可通过git reflog恢复,但有门槛
代码溯源 完整的合并轨迹,可精准定位 bug 是哪个分支、哪次提交引入的,排查问题效率高 提交历史被重写,原始提交的上下文丢失,问题溯源难度大幅提升
公共分支兼容性 完全兼容,是公共分支合并的标准方案 绝对禁止在公共分支使用,会导致团队成员分支历史不一致,引发灾难性冲突

4. 优缺点详解

git merge 优缺点

✅ 优点:

  1. 操作简单、上手门槛低,符合 Git 分布式设计的初衷,全程安全无风险;
  2. 完整保留所有分支的开发上下文和提交历史,方便后续代码审计、问题回溯、版本回滚;
  3. 冲突处理简单,仅需解决一次冲突,不会出现重复处理冲突的情况;
  4. 支持快进合并(Fast-Forward),当目标分支无新提交时,可直接移动分支指针,无需生成合并提交。

❌ 缺点:

  1. 多人协作频繁合并时,会产生大量的合并提交,导致提交历史分叉严重,可读性变差;
  2. 对于追求简洁线性提交历史的团队,多余的合并提交会显得冗余,不利于版本管理。

git rebase 优缺点

✅ 优点:

  1. 最终会形成干净、无分叉的线性提交历史,提交日志可读性极强,方便版本迭代回溯;
  2. 支持交互式变基git rebase -i,可在合并前整理本地提交(合并零散提交、修改提交信息、删除无用提交),让提交记录更规范;
  3. git pull --rebase拉取远程代码,可避免生成多余的合并提交,保持本地分支的线性历史。

❌ 缺点:

  1. 操作风险高,重写提交历史的操作不可逆,一旦失误极易丢失代码;
  2. 冲突处理成本高,多个提交重放时需要逐个解决冲突,重复操作多;
  3. 重写历史后,原始提交的上下文丢失,出现问题时很难精准定位 bug 引入的节点;
  4. 在公共分支使用会给团队带来灾难性后果,所有成员都需要强制同步重写后的历史,极易出现代码丢失、冲突爆炸。

5. 最佳实践 & 使用场景(面试必答,体现你的实战经验)

git merge 推荐使用场景

  1. 将功能分支合并到公共主分支(main/master、develop)时,必须使用 git merge,建议搭配--no-ff参数(禁用快进合并),强制生成合并提交,完整保留分支合并的上下文,方便后续追溯和回滚;
  2. 多人协作的公共分支之间的合并,保证所有团队成员的提交历史一致,不会出现历史混乱;
  3. 合并到上线分支、生产分支时,必须使用 merge,保证所有操作可追溯,出问题可快速回滚;
  4. 需要完整保留分支开发上下文,用于代码审计、合规检查的场景。

git rebase 推荐使用场景

  1. 本地私有功能分支的提交历史整理,比如自己开发的 feature 分支,在合并到公共分支之前,用git rebase -i HEAD~n整理零散的提交,让提交记录语义化、规范化;
  2. 本地分支拉取远程公共分支的最新代码时,用git pull --rebase代替默认的git pull,避免生成多余的合并提交,保持本地分支的线性历史;
  3. 给开源项目提交 PR/MR 时,绝大多数开源项目要求提交历史是线性的,需要用 rebase 基于上游最新分支整理提交,避免合并冲突和冗余的合并提交;
  4. 个人独立开发的项目,想要保持干净的线性提交历史,可自由使用 rebase。

6. 黄金法则(面试答出来直接加分)

永远不要在已经推送到远程的公共分支上,执行 git rebase 操作!

公共分支是所有团队成员的开发基准,你 rebase 之后会重写公共分支的提交历史,其他成员的本地分支还是基于原来的历史,当他们拉取代码时,会出现两个版本的历史,合并后会产生大量重复的提交和无法解决的冲突,最终导致代码仓库的历史彻底混乱,甚至丢失核心代码。


三、前端面试 Git 高频附加题

除了核心的 merge/rebase,这些考点也是面试官常问的,补充在这里帮大家全面通关:

1. git reset 和 git revert 的区别?

核心区别:是否重写提交历史,是否可逆

  • git revert:生成一个新的提交,反向撤销目标提交的所有改动,不会修改现有提交历史,安全,适合公共分支的代码回滚,所有操作可追溯;
  • git reset:直接移动分支指针,删除目标提交之后的所有提交,会重写提交历史,分为--soft(保留改动到暂存区)、--mixed(默认,保留改动到工作区)、--hard(彻底丢弃所有改动,高危),仅适合本地私有分支的回滚,绝对禁止在已推送的公共分支使用。

2. 什么是分离头指针(detached HEAD)?有什么风险?

  • 定义:执行git checkout <提交哈希/标签名>时,HEAD 指针不再指向任何一个命名分支,而是直接指向一个具体的提交记录,此时就进入了分离头指针状态;
  • 风险:此状态下的提交,属于匿名分支上的提交,一旦切换到其他分支,这些提交会被 Git 的垃圾回收机制清理,极易丢失;
  • 解决方案:如果需要在此状态下保留修改,立即执行git checkout -b <新分支名>,创建新分支保存这些提交。

3. .gitignore 文件不生效怎么办?

  • 根本原因:.gitignore只能忽略未被跟踪的文件,如果文件已经被提交到 Git 仓库,就不会被 ignore 规则匹配;

  • 解决方案:

    1. 先把本地需要忽略的文件备份,避免丢失;
    2. 执行git rm -r --cached .,清除所有文件的本地跟踪缓存;
    3. 重新执行git add .,此时.gitignore规则会生效,忽略指定文件;
    4. 提交修改到仓库即可。

4. 不小心把账号密码、密钥等敏感信息提交到 Git 仓库了,怎么办?

  • 第一步:立即修改敏感信息的密码 / 密钥,杜绝泄露风险;

  • 第二步:清理 Git 仓库中的敏感信息:

    • 如果是仅本地提交、未推送到远程:用git reset --soft HEAD~1回滚提交,修改后重新提交即可;
    • 如果已经推送到远程公共仓库:用git filter-repo(官方推荐)或 BFG 工具彻底清理历史记录,清理后需要强制推送重写历史;
  • 第三步:如果是开源仓库,建议联系平台仓库管理员,彻底清理缓存记录。


结尾总结

Git 作为前端开发的必备工具,面试考察的核心从来不是你背了多少命令,而是你是否理解每个操作背后的原理,是否知道不同操作的风险和最佳实践。

核心记住两点:

  1. 公共分支永远用merge,保证安全和可追溯;私有分支可以用rebase整理提交历史,保持简洁;
  2. 所有会重写提交历史的操作(rebasereset --hardcommit --amend),绝对不要用在已经推送到远程的公共分支上。

这篇文章整理了我备战前端实习面试的 Git 核心笔记,希望能帮到同样在找工作的同学。如果觉得有用,欢迎点赞、收藏、评论,我会持续更新前端面试的干货内容~

面试爱问底层时,我是怎么读大型前端源码的❓❓❓

大家好 👋,我是 Moment,目前正在使用 Next.js、NestJS、LangChain 开发 DocFlow。这是一个面向 AI 场景的协同文档平台,集成了基于 Tiptap 的富文本编辑、NestJS 后端服务、实时协作与智能化工作流等核心模块。

在这个项目的持续打磨过程中,我积累了不少实战经验,不只是 Tiptap 的深度定制、编辑器性能优化和协同方案设计,也包括前端工程化建设、React 源码理解以及复杂项目架构实践。

如果你对 AI 全栈开发、文档编辑器、前端工程化或者 React 源码相关内容感兴趣,欢迎添加我的微信 yunmz777 一起交流。觉得项目还不错的话,也欢迎给 DocFlow 点个 star ⭐

image.png

网上类似的源码长文不少,我最近也在写 React 源码。作者往往写得尽兴、覆盖面也广,读者却不一定对得上自己的节奏:你想抠的那一点未必落在文章的主线上,而仓库一直在演进,成稿稍一搁置,对照现版就容易对不上号。

也正因如此,很多同学更倾向于亲自读源码。带着问题找答案,节奏和技术栈都更贴自己。

这篇想分享的是读大型前端开源项目(例如 ReactVueWebpackBabel)源码时怎么切入、怎么少迷路。目标很简单:授人以渔,让你在遇到新机制、底层实现或 Bug 时,能自己钻进去看清楚。

为什么读源码要先有问题

读之前先想清楚:为什么要打开仓库?

我的看法是,首要目的是解决实际问题。没有目标地"逛"仓库,像大海捞针,效率低也容易泄气。反过来,从一个具体问题出发,更容易把设计和实现串起来。

例如你可能会问:为什么在 React 里调用 setState 后,状态不会立刻改掉,而是走一批调度?这个问题会把你带到更新队列和调度相关代码上,比空读文件快得多。

如下图所示。

20260401084026

一图说清这条路径:先有问题,再定入口,沿调用栈往下跟,最后把理解收成自己的模型。

读新版别从第一个 commit 啃起

有人说从第一个 commit 顺着读能看懂演进。对极少数人可行,对大多数人来说性价比很低。以 React 为例,提交量极大,早期设计不少已废弃,啃旧代码对理解当下版本帮助有限。

更稳妥的做法是盯住当前主流版本:社区文章、视频、讨论多,卡住了好搜;API 和你项目里用的是同一世代;可以先读二手资料抓思路,再回仓库对细节。"资料 + 源码"比一行行硬读省时间。等你对现版熟了,再针对某个功能去翻 commit 和 PR,会更有数。

如下图所示。

20260401084233

一图把两种读法摆开:一种以手头在用的主版本为主线,资料帮着搭骨架,演进历史等站稳了再补;另一种是从第一个 commit 起顺序硬啃。取舍在哪,看图就明白。

读源码不是上来就梭哈

React 这种体量的仓库是进阶活。基础不够会越看越懵。建议先具备下面这些块,再往里钻(哪块弱就补哪块,本身也是正经学习)。

  • 语言:ES6+ 常用语法要熟,闭包、原型、异步和事件循环要真的用过,不然 Hooks 和调度相关代码很难读顺。
  • 框架:组件、propsstate、常用 HooksReact 18 里和并发、批更新、Suspense 相关的东西,至少用过再对照源码。
  • 调度直觉:时间片、优先级队列这类概念有个印象即可,读 FiberScheduler 时会轻松些。
  • 基础数据结构:树、链表、堆、 Diff 在干什么,知道个大概即可定位章节。
  • 浏览器与帧:FPSrequestAnimationFrame、为何怕长任务占主线程,有助于理解为什么要切片和中断渲染。

如下图所示。

20260401084431

该备的五块底子和边读边补哪弱补哪,下图一笔带过,正文就不摊开长清单了。

先把源码跑起来再说

第一步不是乱翻文件,而是按 READMECONTRIBUTING.md 把仓库构建起来、能断点。前端框架尽量用本地编出来的 development 包,别拿压缩过的生产包硬读。可以写最小 demo,或用 link 把本地包挂进项目里,贴近真实用法。CONTRIBUTING.md 里往往写了怎么跑测试、怎么编包,这部分本身就是读源码的序章。

如下图所示。

20260401084716

从克隆到能下断点的一圈步骤,对应正文不再逐句展开。

我那边的协同文档 Docflow 里也写了贡献说明和架构笔记,道理一样:先能构建、能跑,再谈读。

理清目录结构再看实现

大仓多是 Monorepo,packages/ 里一块一块职责分明。先当看地图,再进文件,不容易盲人摸象。

以 React 为例,常见分包包括:react(对外 API)、react-dom(对接 DOM)、react-reconciler(调和与更新)、scheduler(优先级与调度)、shared(公共工具)。心里有了这张表,搜到符号时才知道该进哪个包。

如下图所示。

20260401084924

React 各包各管一摊,读源码时先认准该进哪个包再翻文件,下面的版式把分工和这个习惯叠在一眼能扫开的地方。

如何调试 React 源码

调试前要会编开发包。示例流程:git clone React 仓库,yarn install,再 yarn build react/index,react-dom/index --type=NODE(或需要浏览器时用 --type=UMD_DEV)。更贴近日常的做法是建一个小项目,用 yarn link 把本地构建产物链进去。

搭建调试环境

具体命令随仓库文档变动,以官方 CONTRIBUTING.md 为准。下面只记思路:依赖装好、开发包产出、demo 或 link 接上。

如下图所示。

20260401085429

构建与 link、再在浏览器里下的那一套,和正文里的命令说明互补。

调试 useState 的执行流程

在业务组件里写一个最小 useState 示例。源码里对外声明多在 packages/react/src/ReactHooks.js,实现落在 packages/react-reconciler/src/ReactFiberHooks.jsmountStateupdateState。在几处入口加 debugger,重建后刷新页面,看调用栈:useStatedispatcher.useStatemountStateupdateState,再单步看链表与更新对象如何挂到 fiber 上。

调试 useEffect 的执行时机

useEffect 跨阶段更多:在 ReactFiberHooks.js 里看 mountEffectupdateEffect 如何在 render 阶段挂 effect,再到 ReactFiberCommitWork.js 里跟 commitLayoutEffectsflushPassiveEffects,能看清 passive 为何在布局后异步跑、为何不挡绘制。

调试技巧与注意事项

频繁触发的路径用条件断点(例如只在某个 fiber.type.name 上停)。if (__DEV__) 和纯告警逻辑可先跳过。Call StackScopeWatch 里盯住 fiber.memoizedStatefiber.updateQueue 等字段。双缓存时要分清当前在 current 还是 workInProgress 上,可配合 fiber.alternate 对照。

debugger 与全局搜索一起用

问题驱动的一个完整例子:搞清楚类组件里调用 setState 之后内部大致怎么走。先用全局搜索在 packages/react/src/ReactBaseClasses.js 找到入口,在本地加断点(下例仅示意,与仓库真实实现一致处请以你检出版本为准)。

// 示意:类组件 setState 入口会委托 updater(真实代码以仓库为准)
Component.prototype.setState = function (
  partialState: object | ((prev: any, props: any) => object) | null,
  callback?: () => void,
): void {
  this.updater.enqueueSetState(this, partialState, callback, "setState");
};

触发断点后跟栈,会进入 react-reconciler 里的 enqueueSetState、更新入队,再到 scheduleUpdateOnFiber 一类调度入口。下面用 Mermaid 收束主链路,细节仍靠你在关键函数上停。

20260401085625

Performance 面板适合观察并发下任务切片、长任务是否让出主线程;和源码里的时间片策略对照着看,比纯文字描述直观。

断点不要从入口无脑单步。beginWorkcompleteWorkcommitRoot、各生命周期与 Hooks 关键函数,按问题选挂。Node 工具链则多看插件注册与钩子调用处。主流程外的 __DEV__ 分支、冗长报错拼装,知道存在即可,不必逐行啃。

宏观上,React 一次更新可以粗分为 render(生成或调和 Fiber 树)与 commit(提交到 DOM)。render 里又可记 beginWork 向下、completeWork 向上;commit 里再分 beforeMutationmutationlayout 等子阶段。先记住这张骨架,再按需钻 reconcileChildrenflushPassiveEffects 之类细节。

如下图所示。

20260401085840

render 与 commit 的分段记忆图,和上面的 Mermaid 互补。

官方一手资料别浪费

维护者在博客、GitHub、演讲里解释"为什么这样设计"的句子,往往比第三手摘要靠谱。Issue 里长讨论、RFC 仓库里的提案与反对意见,都是源码的"旁白",代码告诉你是什么,这些文字告诉你为什么。按关键词搜 schedulerFiber、你关心的特性名,常能挖到设计取舍。

如下图所示。

20260401090038

博客、演讲、Issue、PR、RFC、发布说明,六类一手材料与"代码加为什么"的对照。

借助大模型但要自己验证

把难读的片段贴给模型,请它讲控制流和字段含义,能省大量初读时间。长 Issue、RFC 可先让模型摘要,再挑段落精读。仓库级助手(例如 Copilot 一类)适合问"谁调用了这个符号"。输出要当草稿,和本地断点、官方文档交叉验证,思维模型还是要自己搭。

如下图所示。

20260401090157

下面六道顺手用法之外,单独压一条硬底线:模型说得再顺,也要用本地断点和官方文档对一遍,不必在正文里一条条摊开。

总结

读大型源码,不必把全文再背一遍,抓住几条习惯就够把前面的方法串起来。

带着问题进门,版本对准你日常在用的主线,基础薄就先补。仓库能构建、能下断点,再谈细读。packages 当地图,先认包再走文件。debugger 配合全局搜索沿调用栈往下跟,官方讨论、Issue、RFC、发布说明补上代码里看不见的为什么。大模型可以加速梳理,最后一步仍要落回本机跑一遍,和官方文档对上。

如下图所示。

20260401090432

习惯之间的层次和先后,用上面这张比把前文再拉长复述更省事。

源码不玄,只是别人把取舍写进了可运行的形式里。节奏对了,会越读越轻。别指望一次吃透,那是慢功夫。路径熟了,换一套框架也能沿用同一套钻法。

🔥前端跨域封神解法:Vite Proxy + Express CORS,一篇搞定所有跨域坑!

🔥前端跨域封神解法:Vite Proxy + Express CORS,一篇搞定所有跨域坑!

全网最通俗跨域教程|前端 Vue/React 通用|后端仅 Express|开发 / 生产全覆盖

前言

做前端开发,跨域绝对是新手最崩溃的拦路虎!浏览器同源策略一拦,接口请求直接报错 No 'Access-Control-Allow-Origin',调试半天毫无头绪。

今天直接给你两套绝杀方案,全程只用到 Vite 代理 + Express 后端:✅ 本地开发用 Vite Proxy 代理(零后端改动,秒解决)✅ 线上生产用 Express CORS 配置(标准规范,永久生效)一文吃透,从此跨域再也不是问题!


一、先搞懂:到底什么是跨域?

浏览器同源策略:协议、域名、端口任意一个不同,就是跨域

举个例子:

  • 前端:http://localhost:5173(Vite 默认端口)
  • 后端:http://localhost:3000(Express 服务)端口不同 → 直接跨域,接口被浏览器拦截!

典型跨域报错:No 'Access-Control-Allow-Origin' header is present on the requested resource.


二、方案 1:本地开发神器 ✨ Vite Proxy 代理

核心原理

前端不直接请求后端,交给Vite 开发服务器做中间人转发,绕过浏览器同源限制,纯前端配置,后端零改动

完整配置(Vue / React 二选一)

1. Vue 版本
// vite.config.js
import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'

export default defineConfig({
  // Vue 编译插件
  plugins: [vue()],
  // 开发服务器配置
  server: {
    // 跨域代理核心配置
    proxy: {
      // 匹配所有 /api 开头的接口
      '/api': {
        target: 'http://localhost:3000', // Express 后端真实地址
        changeOrigin: true, // 🔥 关键:伪装来源,解决跨域
        pathRewrite: {
          '^/api': '' // 路径重写,前端 /api/login → 后端 /login
        }
      }
    }
  }
})
2. React 版本
// vite.config.js
import { defineConfig } from 'vite'
import react from '@vitejs/plugin-react'

export default defineConfig({
  // React 编译插件
  plugins: [react()],
  // 代理配置和 Vue 完全一致!
  server: {
    proxy: {
      '/api': {
        target: 'http://localhost:3000',
        changeOrigin: true,
        pathRewrite: {
          '^/api': ''
        }
      }
    }
  }
})

关键配置解读

  • target:Express 后端接口真实地址
  • changeOrigin: true:伪装请求来源,让后端认为是同源请求
  • pathRewrite:路径重写,简化前端接口书写

适用场景

仅限本地开发环境上线打包后代理失效,生产环境必须用 CORS!


三、方案 2:生产环境标配 🚀 Express CORS 配置

核心原理

后端在响应头中添加跨域允许规则,明确告诉浏览器:允许这个前端域名访问我的接口。

需要配置的三个核心响应头:

Access-Control-Allow-Origin: 允许的前端域名
Access-Control-Allow-Methods: 允许的请求方法
Access-Control-Allow-Headers: 允许的请求头

完整 Express 配置(直接复制可用)

// 1. 初始化项目:npm init -y
// 2. 安装依赖:npm install express cors
const express = require('express')
const cors = require('cors')
const app = express()

// 解析 JSON 请求体
app.use(express.json())

// 🔥 CORS 核心配置(生产环境必写)
app.use(cors({
  // 允许访问的前端域名(本地开发/线上替换即可)
  origin: 'http://localhost:5173',
  // 允许的请求方式
  methods: ['GET', 'POST'],
  // 允许的请求头
  allowedHeaders: ['Content-Type'],
  // 允许携带Cookie(登录场景必开)
  credentials: true
}))

// 测试接口
app.get('/user', (req, res) => {
  res.send({ 
    code: 200, 
    msg: '请求成功',
    data: { name: '前端开发者' } 
  })
})

// 启动 Express 服务
app.listen(3000, () => {
  console.log('Express 服务启动:http://localhost:3000')
})

极简原生写法(不依赖 cors 包)

如果不想安装第三方包,直接手动设置响应头:

const express = require('express')
const app = express()
app.use(express.json())

// 手动配置 CORS 响应头
app.use((req, res, next) => {
  res.header('Access-Control-Allow-Origin', 'http://localhost:5173')
  res.header('Access-Control-Allow-Methods', 'GET,POST')
  res.header('Access-Control-Allow-Headers', 'Content-Type')
  res.header('Access-Control-Allow-Credentials', 'true')
  next()
})

// 接口
app.get('/user', (req, res) => {
  res.send({ code: 200, msg: '请求成功' })
})

app.listen(3000)

适用场景

生产环境正式上线Express 专属标准解决方案,全网通用。


四、Proxy vs CORS 到底怎么选?

表格

方案 适用环境 优点 缺点
Vite Proxy 本地开发 零后端改动,配置简单 上线失效
Express CORS 生产环境 标准规范,永久生效 需要后端配置

最佳实践开发用 Proxy,上线用 CORS,两套方案无缝衔接!


五、高频踩坑总结

  1. changeOrigin: true 忘记写 → 跨域依然报错
  2. 路径重写错误 → 接口 404
  3. CORS 域名配置错误 → 线上依然跨域
  4. 开发 / 生产配置混用 → 线上接口异常
  5. 请求方式超出允许范围 → 预检请求失败

结语

跨域根本不是难题,只是没找对方法!Proxy 搞定开发,CORS 搞定生产,照着这篇配置,从此和跨域报错说拜拜~

需要完整 Demo 源码的小伙伴,评论区扣「跨域」直接发你!

💡 关注我,持续输出前端硬核干货,Vue/React/Express 一站式学习!

❌