ReactJS 中的 JSX工作原理
文章目录
- 前言
- ✅ 1. JSX 是什么?
- 🔧 2. 编译后的样子(核心机制)
- 🧱 3. `React.createElement` 做了什么?
- 🧠 4. JSX 与组件的关系
- 🔄 5. JSX 到真实 DOM 的过程
- 📘 6. JSX 与 Fragment、条件渲染等底层行为
- ⚠️ 7. 注意事项
- ✅ 总结
- JSX 编译后如何通过 Fiber 架构驱动渲染更新
- ✅ 一、React 渲染整体流程概览(React 18+)
- 🧠 二、什么是 Fiber 架构?
- 📦 一个 Fiber 节点结构简化如下:
- 🔄 三、Reconciliation(协调 / diff)机制
- Diff 规则核心简化:
- ⚙️ 四、Commit 阶段(更新真实 DOM)
- ⏱️ 五、Concurrent Mode(React 18)
- 🌟 调度示意(非阻塞渲染):
- 📌 六、双缓冲机制(双 Fiber Tree)
- 🎯 七、性能优化的启示
- ✅ 总结:JSX 到 DOM 的底层机制主线
- ⏱️ React Fiber 调度与中断机制:时间切片、优先级调度、任务恢复
- ✅ 1. 为什么需要中断渲染?
- 问题:
- ✅ 2. Fiber 如何解决这个问题?
- 🎯 利用:
- 📦 3. 任务调度:Scheduler 的优先级系统
- ⏳ 4. 时间切片机制(Time Slicing)
- 🔁 5. 中断与恢复机制
- ⚙️ 6. Concurrent 模式的核心优势
- 🧠 7. 示例:startTransition 与中断渲染
- ✅ 8. 总结:Fiber 调度机制的精髓
- 📘 延伸(可选深入)
前言
ReactJS 中的 JSX(JavaScript XML)看起来像 HTML,但它其实只是语法糖,在底层是被编译成 React.createElement(...)
调用的 JavaScript 代码。理解其底层机制对优化性能、调试问题、使用 Hooks 等非常重要。
✅ 1. JSX 是什么?
JSX 是 JavaScript 的语法扩展,允许你这样写 UI:
const element = <h1>Hello, world!</h1>;
它看起来像 HTML,但不是。它不会被浏览器直接执行,而是会在构建阶段被转译为 JavaScript。
🔧 2. 编译后的样子(核心机制)
这段代码:
const element = <h1 className="title">Hello</h1>;
会被 Babel 编译为:
const element = React.createElement('h1',{ className: 'title' },'Hello'
);
等价于:
{type: 'h1',props: {className: 'title',children: 'Hello'}
}
最终,React 用这些对象来构建虚拟 DOM(Virtual DOM)。
🧱 3. React.createElement
做了什么?
它会返回一个React 元素对象:
{$$typeof: Symbol(react.element),type: 'h1',key: null,ref: null,props: {className: 'title',children: 'Hello'},_owner: null
}
这是一个纯 JS 对象,描述了 UI 的结构。不是 DOM 节点,还没有真正渲染。
🧠 4. JSX 与组件的关系
JSX 中也可以使用组件:
<MyButton text="Click me" />
被编译为:
React.createElement(MyButton, { text: "Click me" });
最终调用你自己写的 MyButton(props)
函数,返回另一个 createElement(...)
。
🔄 5. JSX 到真实 DOM 的过程
- JSX 被编译为
React.createElement(...)
- React 构建虚拟 DOM 树
- React 将虚拟 DOM diff(比对)旧的 DOM 树
- React 使用 DOM API 更新真实 DOM(插入、删除、替换节点)
📘 6. JSX 与 Fragment、条件渲染等底层行为
<></>
→React.createElement(React.Fragment, null, ...)
- 条件渲染:
{condition && <div>Yes</div>}
→ 三元或逻辑判断 - 循环渲染:
{list.map(item => <li>{item}</li>)}
→ 多次 createElement 调用
⚠️ 7. 注意事项
特性 | 底层意义 |
---|---|
JSX 只能有一个根节点 | 因为 React.createElement 只能返回一个元素 |
不支持 if 语句直接写在 JSX | JSX 是表达式,需要用三元或外部逻辑 |
自闭和标签 <MyComp /> | 转译为 React.createElement(MyComp) |
不能直接渲染对象 | [object Object] 错误是因为 createElement 接收了非法 children |
✅ 总结
概念 | 作用 |
---|---|
JSX | React 的语法糖,用来描述 UI 结构 |
Babel | 把 JSX 编译为 React.createElement() |
React.createElement | 创建虚拟 DOM 描述对象 |
虚拟 DOM | 一个 JS 对象树,最终映射为真实 DOM |
渲染机制 | Diff + 更新 DOM 节点(最小化操作) |
JSX 编译后如何通过 Fiber 架构驱动渲染更新
这一主线,系统讲解 React 的底层工作机制,包括:
✅ 一、React 渲染整体流程概览(React 18+)
graph TD
A[JSX] --> B[createElement]
B --> C[构建虚拟 DOM]
C --> D[Fiber 架构构建 Fiber Tree]
D --> E[Reconciliation(diff)]
E --> F[Commit 阶段:更新真实 DOM]
🧠 二、什么是 Fiber 架构?
Fiber 是 React 从 v16 开始的核心架构,其主要目标是:
- 支持任务中断(可暂停)
- 支持异步渲染(并发模式)
- 增强调度控制
- 支持优先级(Urgent、Normal、Idle)
Fiber = 一种“工作单元”结构,用链表组织虚拟 DOM,便于逐个处理、可打断、可恢复
📦 一个 Fiber 节点结构简化如下:
interface Fiber {type: string | FunctionComponentstateNode: HTMLElement | nullchild: Fiber | nullsibling: Fiber | nullreturn: Fiber | nullalternate: Fiber | null // 双缓存机制effectTag: 'UPDATE' | 'PLACEMENT' | 'DELETION'
}
🔄 三、Reconciliation(协调 / diff)机制
目的:比较新旧 virtual DOM,找到最小变更,生成 effectList 用于更新真实 DOM。
Diff 规则核心简化:
- 同层比较:只比较同级节点
- Key 区分同类型组件:用于
map
渲染性能优化 - 类型不同 → 全替换
- 类型相同 → 尝试复用
⚙️ 四、Commit 阶段(更新真实 DOM)
当 Fiber 树构建完毕后,进入 commit 阶段:
// commitWork
switch (fiber.effectTag) {case 'PLACEMENT': // 插入parent.appendChild(fiber.stateNode);break;case 'UPDATE': // 属性或内容变更updateDOM(fiber.stateNode, oldProps, newProps);break;case 'DELETION':parent.removeChild(fiber.stateNode);break;
}
⏱️ 五、Concurrent Mode(React 18)
React 18 引入并发渲染模式,依赖 Fiber 实现:
特性 | 描述 |
---|---|
startTransition | 标记可中断更新(非紧急) |
自动批处理 | setState 自动分组,不再需要 unstable_batchedUpdates |
useDeferredValue | 延迟非关键状态同步更新 |
useTransition | 标记 UI “pending” 状态,支持 loading skeleton 等过渡体验 |
ReactDOM.createRoot | 默认进入 Concurrent 模式 |
🌟 调度示意(非阻塞渲染):
startTransition(() => {setValue(input);
})
React 会调度优先级高的操作(如输入)先渲染,低优先的任务稍后处理。
📌 六、双缓冲机制(双 Fiber Tree)
React 使用 current
和 workInProgress
两棵 Fiber Tree,交替使用:
current | 正在显示的 UI |
---|---|
workInProgress | 当前计算的新状态 |
当 Reconciliation 完成,React 会“交换”两棵树,实现 快照切换。
🎯 七、性能优化的启示
场景 | 原因 | 优化策略 |
---|---|---|
大量列表重复渲染 | Key 设计不当导致全部 diff | 用稳定 Key |
卡顿严重 | 所有状态更新同步 | 使用 startTransition , useDeferredValue |
多组件嵌套,重渲染过多 | props 未 memo | 使用 React.memo 、useMemo , useCallback |
复杂条件渲染 | JSX 重计算多 | 使用懒加载 React.lazy , Suspense |
✅ 总结:JSX 到 DOM 的底层机制主线
阶段 | 关键内容 |
---|---|
编译 | JSX → createElement → 虚拟 DOM |
构建 | Fiber 树结构,链表连接 |
diff | Reconciliation 识别变化 |
更新 | commit 阶段执行 effect list 操作真实 DOM |
并发 | Fiber 允许中断、恢复、优先级调度 |
优化 | 利用 Hook + memo + key + transition |
⏱️ React Fiber 调度与中断机制:时间切片、优先级调度、任务恢复
React Fiber 架构最大的突破之一就是它将渲染任务拆分为“工作单元”(Fiber 节点),并使用浏览器的空闲时间或非阻塞方式来执行这些任务。
✅ 1. 为什么需要中断渲染?
问题:
React 早期(v15 及以前)渲染是同步的:
ReactDOM.render(<App />, root);
如果组件树很大,JS 线程会被“卡死”,导致:
- 输入卡顿
- 动画掉帧
- 无法响应用户操作
✅ 2. Fiber 如何解决这个问题?
🧠 核心思想:把渲染拆成很多小任务,用浏览器空闲时间分批执行,并可中途暂停与恢复。
🎯 利用:
requestIdleCallback(callback)
或(React 18 起)用:
scheduler.unstable_scheduleCallback(priority, callback)
📦 3. 任务调度:Scheduler 的优先级系统
React 使用内部调度器(scheduler
)给任务分配优先级:
调度级别 | 描述 | 示例 |
---|---|---|
Immediate (最高) | 立刻执行 | setState 同步 |
UserBlocking | 用户操作相关 | 输入、点击 |
Normal | 默认更新 | 内容变化 |
Low | 非关键 | 动画、日志 |
Idle | 不重要 | 预加载 |
⏳ 4. 时间切片机制(Time Slicing)
Fiber 会把任务拆成一个个 unit of work,然后循环调度:
while (nextUnitOfWork && shouldYield() === false) {nextUnitOfWork = performUnitOfWork(nextUnitOfWork);
}
shouldYield()
检查是否要让位给浏览器,防止掉帧。
function shouldYield() {return performance.now() >= deadline;
}
🔁 5. 中断与恢复机制
如果 shouldYield()
为 true
,Fiber 会 中断当前工作,然后把当前状态保存在 workInProgress
上,下次从这个点继续:
// workInProgress 保存当前 fiber 的指针链表状态
浏览器空闲后,React 会再次调度:
requestIdleCallback(workLoop)
⚙️ 6. Concurrent 模式的核心优势
能力 | 说明 |
---|---|
✋ 中断 | 渲染过程可被暂停 |
🔁 恢复 | 下一帧继续渲染剩余部分 |
⚖️ 优先级 | 用户输入优先渲染,非关键可延后 |
🎨 更流畅 | 输入不卡顿,动画更自然 |
🧠 更智能 | 可实现并行 diff、预渲染、Suspense fallback 等功能 |
🧠 7. 示例:startTransition 与中断渲染
import { startTransition } from 'react'function handleInput(e) {const value = e.target.value// 非紧急更新,允许中断startTransition(() => {setFilteredList(filterBigList(value))})
}
用户打字不会因为
setFilteredList()
而卡顿,因为这是一个“可中断更新”。
✅ 8. 总结:Fiber 调度机制的精髓
点 | 内容 |
---|---|
拆分 | Fiber 把渲染任务拆成小单元 |
调度 | 使用 scheduler 或 requestIdleCallback 进行执行调度 |
中断 | shouldYield() 决定是否让步给浏览器主线程 |
恢复 | 保存中间状态于 Fiber,支持恢复渲染 |
优先级 | 按任务重要程度进行排序调度,保障交互流畅性 |
📘 延伸(可选深入)
如果你还想进一步了解:
Lane Model
(React 18 调度核心)- React Scheduler 实现(源码级)
- Suspense + Fiber 如何协同暂停渲染树
Concurrent Features
:如React.lazy
,Streaming SSR
相关文章:
ReactJS 中的 JSX工作原理
文章目录 前言✅ 1. JSX 是什么?🔧 2. 编译后的样子(核心机制)🧱 3. React.createElement 做了什么?🧠 4. JSX 与组件的关系🔄 5. JSX 到真实 DOM 的过程📘 6. JSX 与 Fr…...

《STL--stack 和 queue 的使用及其底层实现》
引言: 上次我们学习了容器list的使用及其底层实现,相对来说是比较复杂的,今天我们要学习的适配器stack和queue与list相比就简单很多了,下面我们就开始今天的学习: 一:stack(后进先出ÿ…...
ArcGIS Pro 3.4 二次开发 - 地理处理
环境:ArcGIS Pro SDK 3.4 + .NET 8 文章目录 地理处理1 通用1.1 如何执行模型工具1.2 设置地理处理范围环境1.3 在 Geoprocessing 窗格中打开脚本工具对话框1.4 打开特定工具的地理处理工具窗格1.5 获取地理处理项目项1.6 阻止通过GP创建的特征类自动添加到地图中1.7 GPExecut…...

基于springboot的医护人员排班系统设计与实现(源码+文档+部署讲解)
技术范围:SpringBoot、Vue、SSM、HLMT、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、小程序、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容:免费功能设计、开题报告、任务书、中期检查PPT、系统功能实现、代码编写、论文编写和辅导、论文…...
Asp.Net Core FluentValidation校验框架
文章目录 前言一、使用步骤1.安装 NuGet 包2.创建模型3.创建验证器4.配置 Program.cs5.创建控制器6.测试结果 二、常见问题及注意事项三、性能优化建议总结 前言 FluentValidation 是一个流行的 .NET 库,用于构建强类型的验证规则。它通常用于验证领域模型、DTO等对…...

CRISPR-Cas系统的小型化研究进展-文献精读137
Progress in the miniaturization of CRISPR-Cas systems CRISPR-Cas系统的小型化研究进展 摘要 CRISPR-Cas基因编辑技术由于其简便性和高效性,已被广泛应用于生物学、医学、农学等领域的基础与应用研究。目前广泛使用的Cas核酸酶均具有较大的分子量(通…...

利用python工具you-get下载网页的视频文件
有时候我们可能在一个网站看到一个视频(比如B站),想下载,但是页面没有下载视频的按钮。这时候,我们可以借助python工具you-get来实现下载功能。下面简要说下步骤 (一)因为使用的是python工具&a…...
Wi-Fi 切换 5G 的时机
每天都希望 Wi-Fi 在我离开信号覆盖范围时能尽快切到 5G,但每次它都能坚挺到最后半格信号,我却连看个天气预报都看不了…我不得不手工关闭 Wi-Fi,然后等走远了之后再打开,如此反复,不厌其烦。 早上出门上班,…...
【请关注】各类数据库优化,抓大重点整改,快速优化空间mysql,Oracle,Neo4j等
各类数据库优化,抓大重点整改,快速优化,首先分析各数据库查询全部表的空间大小及记录条数的语句: MySQL -- 查看所有表的空间大小 SELECT TABLE_SCHEMA AS 数据库名, TABLE_NAME AS 表名, ENGINE AS 存储引擎, CONCAT(ROUND(DAT…...
Mybatis Plus JSqlParser解析sql语句及JSqlParser安装步骤
MyBatis Plus 整合 JSqlParser 进行 SQL 解析的实现方案,主要包括环境配置和具体应用。通过 Maven 添加mybatis-plus-core 和 jsqlparser 依赖后,可用 CCJSqlParserUtil 解析 SQL 语句,支持对 SELECT、UPDATE 等语句的语法树分析和重构。技术…...
React从基础入门到高级实战:React 高级主题 - 性能优化:深入探索与实践指南
React 性能优化:深入探索与实践指南 引言 在现代Web开发中,尤其是2025年的技术环境下,React应用的性能优化已成为开发者不可忽视的核心课题。随着用户对应用速度和体验的要求日益提高,React应用的规模和复杂性不断增加ÿ…...
负载均衡群集---Haproxy
目录 一、HAproxy 一、概念 二、核心作用 三、主要功能特性 四、应用场景 五、优势与特点 二、 案例分析 1. 案例概述 2. 案例前置知识点 (1)HTTP 请求 (2)负载均衡常用调度算法 (3)常见的 web …...
2025年5月个人工作生活总结
本文为 2025年5月工作生活总结。 研发编码 一个项目的临时记录 月初和另一项目同事向业主汇报方案,两个项目都不满意,后来领导做了调整,将项目合并,拆分了好几大块。原来我做的一些工作,如数据库、中间件等ÿ…...

【stm32开发板】单片机最小系统原理图设计
一、批量添加网络标签 可以选择浮动工具中的N,单独为引脚添加网络标签。 当芯片引脚非常多的时候,选中芯片,右键选择扇出网络标签/非连接标识 按住ctrl键即可选中多个引脚 点击将引脚名称填入网络名 就完成了引脚标签的批量添加 二、电源引…...

实验设计与分析(第6版,Montgomery)第5章析因设计引导5.7节思考题5.2 R语言解题
本文是实验设计与分析(第6版,Montgomery著,傅珏生译) 第5章析因设计引导5.7节思考题5.2 R语言解题。主要涉及方差分析,正态假设检验,残差分析,交互作用。 dataframe<-data.frame( Surfacec(74,64,60,92…...

2025山东CCPC题解
文章目录 L - StellaD - Distributed SystemI - Square PuzzleE - Greatest Common DivisorG - Assembly Line L - Stella 题目来源:L - Stella 解题思路 签到题,因为给出的字母不是按顺序,可以存起来赋其值,然后在比较。 代码…...
【解决办法】ubuntu重启不起来,输入用户名和密码进不去,又重新返回登录页。
项目场景: ubuntu重启不起来,输入用户名和密码进不去,又重新返回登录页。 问题描述 在华硕天选一代笔记本上面安装了ubuntu22.04.5桌面版,但是重启以后出现,输入了用户名和密码,等待一会还让输入用户名和…...

CentOS Stream 9 中部署 MySQL 8.0 MGR(MySQL Group Replication)一主两从高可用集群
🐇明明跟你说过:个人主页 🏅个人专栏:《MySQL技术精粹》🏅 🔖行路有良友,便是天堂🔖 目录 一、前言 1、MySQL 8.0 中的高可用方案 2、适用场景 二、环境准备 1、系统环境说明…...

pycharm 新UI 固定菜单栏 pycharm2025 中文版
pycharm 新UI 文件 -> 设置 -> 外观与行为 -> 外观 -> UI选项 -> 主菜单:显示在主工具栏上方. 即可固定...
跟单业务和量化交易业务所涉及到的设计模式
🔁 跟单业务中常用的设计模式: 1. 观察者模式(Observer) 场景:一个大V下单,系统需要自动通知所有跟随者进行同步下单。好处:解耦下单者与跟随者,支持灵活扩展、异步通知。面试亮点…...

我的世界Java版1.21.4的Fabric模组开发教程(十一)创建方块
这是适用于Minecraft Java版1.21.4的Fabric模组开发系列教程专栏第十一章——创建方块。想要阅读其他内容,请查看或订阅上面的专栏。 方块(Block) 是构成Minecraft世界的主要组成部分,是组成游戏地图的最基本单元,也是模组开发的核心元素之一…...

VR/AR 视网膜级显示破局:10000PPI 如何终结颗粒感时代?
一、传统液晶 “纱窗效应”:VR 沉浸体验的最大绊脚石 当用户首次戴上 VR 头显时,眼前密密麻麻的像素网格往往打破沉浸感 —— 这正是传统液晶显示在近眼场景下的致命缺陷。受限于 500-600PPI 的像素密度,即使达到 4K 分辨率,等效到…...
C++ 命令模式:设计与实现详解
一、引言 在软件开发中,我们经常需要将“请求”或“操作”封装成对象,以便在不同的上下文环境中传递、存储、延迟执行或撤销。命令模式(Command Pattern)正是为解决这类问题而生的行为设计模式。本文将深入探讨 C++ 中命令模式的设计理念、实现方式及其应用场景。 二、命…...

系统思考:化繁为简的艺术
系统思考,其实是一门化繁为简的艺术。当我们能够把复杂的问题拆解成清晰的核心以及更加简单,从而提升团队的思考品质和行动品质,发挥最大的合力。 每个公司都想在某方面成为最优秀的,但是实际上具有穿透性的洞察力和摆脱虚荣心的清…...
java/mysql/ES下的日期类型分析
mysql的timestamp和datetime mysql的TIMESTAMP类型内部存的是unix时间戳,可认为是一个32位的整型,它记录了1970.1.1以来的秒数。因为存储长度4字节的限制,所以有2038年限制。 DATETIME类型内部存的是long型,记录了1000.1.1以来的…...

Angularjs-Hello
1 关于Angularjs 最近因为项目需要又要做这个,所以简单复习下。其实这个大概7,8年前就用过,当时做了几个简单页面觉得太简单就还是回去做嵌入式了。按照互联网技术的进化速度,本来以为早死在 沙滩上了,没想到现在还在坚…...
Python训练营---Day41
DAY 41 简单CNN 知识回顾 数据增强卷积神经网络定义的写法batch归一化:调整一个批次的分布,常用与图像数据特征图:只有卷积操作输出的才叫特征图调度器:直接修改基础学习率 卷积操作常见流程如下: 1. 输入 → 卷积层 …...

Linux 1.0.4
父子shell linux研究的就是shell 打开两个窗口就是两个shell 终端的软件有很多 bash也是一个软件 我们在terminal里面再打开一个bash,然后再次使用ps命令发现多出来一个bash,之后点击exit只是显示了一个exit,这个只是退出了在terminal中打开…...

Qt -下载Qt6与OpenCV
博客主页:【夜泉_ly】 本文专栏:【暂无】 欢迎点赞👍收藏⭐关注❤️ 前言 呃啊,本来就想在 Qt 里简单几个 OpenVC 的函数,没想到一搞就是一天。 我之前的开发环境是 Qt 5.14.2,使用 MinGW 7.3.0 64-bit 编…...

机器学习无监督学习sklearn实战一:K-Means 算法聚类对葡萄酒数据集进行聚类分析和可视化( 主成分分析PCA特征降维)
本项目代码在个人github链接:https://github.com/KLWU07/Machine-learning-Project-practice/tree/main/1-Wine%20cluster%20analysis 如果对于聚类算法理论不理解可参考这篇之前文章机器学习中无监督学习方法的聚类:划分式聚类、层次聚类、密度聚类&…...