React源码解析(1): JSX语法与react项目渲染过程
好家伙
0.前言
由于工作的需要,我不得不入手了react的全家桶,曾经我的主要技术栈是vue。
从vue转到react,一开始我感到非常不适应,jsx的语法的不了解,react hooks的使用方式,react路由的配置。。。这一度让我十分难受
但在熟悉一段时间后,我逐渐领略到react的魅力,灵活的状态管理,渲染速度,与vite集成后超快的打包速度,以及超漂亮UI组件库NextUI
(这真是我目前为止遇到的最漂亮的一套组件库,圆角,配色,动画交互)
让我们拥抱新技术吧!
1. 关于 JSX
JSX 就是“JavaScript + XML”的神奇结合。它让我们能在写 JavaScript 的同时写出类似 XML 标签的东西,看起来就像是这样:
function App() {
return (
<NextUIProvider>
<RouterProvider router={router} />
</NextUIProvider>
);
}
实际上,这些标签最后会被编译成纯 JavaScript,也就是 React 官方的说法——JSX 会被转换成 React.createElement()。
去Babel的网站试一下吧
import React from "react"; export function StudentTable() {
const students = ["panghu", "xiaofu", "daxiong", "jinxiang"]; return (
<table style={{ borderCollapse: "collapse", width: "50%", margin: "1rem auto" }}>
<thead>
<tr style={{ backgroundColor: "#f2f2f2" }}>
<th style={{ border: "1px solid #ccc", padding: "8px" }}>序号</th>
<th style={{ border: "1px solid #ccc", padding: "8px" }}>姓名</th>
</tr>
</thead>
<tbody>
{students.map((name, index) => (
<tr key={index}>
<td style={{ border: "1px solid #ccc", padding: "8px" }}>{index + 1}</td>
<td style={{ border: "1px solid #ccc", padding: "8px" }}>{name}</td>
</tr>
))}
</tbody>
</table>
);
}
Babel编译后

但用 JSX 可读性更好,写着也舒爽。
2.为什么要用jsx?
我们来看vue和react实现相同表格
React
import React from "react"; export function StudentTable() {
const students = ["panghu", "xiaofu", "daxiong", "jinxiang"]; return (
<table style={{ borderCollapse: "collapse", width: "50%", margin: "1rem auto" }}>
<thead>
<tr style={{ backgroundColor: "#f2f2f2" }}>
<th style={{ border: "1px solid #ccc", padding: "8px" }}>序号</th>
<th style={{ border: "1px solid #ccc", padding: "8px" }}>姓名</th>
</tr>
</thead>
<tbody>
{students.map((name, index) => (
<tr key={index}>
<td style={{ border: "1px solid #ccc", padding: "8px" }}>{index + 1}</td>
<td style={{ border: "1px solid #ccc", padding: "8px" }}>{name}</td>
</tr>
))}
</tbody>
</table>
);
}
Vue
<template>
<table style="border-collapse: collapse; width: 50%; margin: 1rem auto;">
<thead>
<tr style="background-color: #f2f2f2;">
<th style="border: 1px solid #ccc; padding: 8px;">序号</th>
<th style="border: 1px solid #ccc; padding: 8px;">姓名</th>
</tr>
</thead>
<tbody>
<tr
v-for="(student, index) in students"
:key="index"
>
<td style="border: 1px solid #ccc; padding: 8px;">{{ index + 1 }}</td>
<td style="border: 1px solid #ccc; padding: 8px;">{{ student }}</td>
</tr>
</tbody>
</table>
</template> <script>
export default {
name: "StudentTable",
data() {
return {
students: ["panghu", "xiaofu", "daxiong", "jinxiang"]
};
}
};
</script>
两段代码对比下来,你会发现
JSX语法虽然对UI进行了描述,但是,将“模板”和“逻辑”混合在一起了
且在单文件内,就可以实现多组件,还不用考虑作用域,真是太爽了!
3.react项目渲染过程
3.1.ReactElement 的数据结构
3.2.渲染流程
- 开发者使用 JSX 语法编写组件,这种语法类似于 HTML,但可以在 JavaScript 中使用。
- Babel 编译:
- Babel 是一个 JavaScript 编译器,它将 JSX 转换为 React.createElement 调用。这一步将 JSX 语法转化为 JavaScript 代码,使其可以在浏览器中运行。
3. React.createElement 调用:
- React.createElement 是一个函数,用于创建 React 元素(ReactElement)。每个 JSX 标签都会被转换为一个 React.createElement 调用。
4. ReactElement 调用:
- ReactElement 是一个轻量级的描述对象,表示界面上某个节点的结构和属性。它是构建虚拟 DOM 的基础。
5. 虚拟 DOM:
- React 使用虚拟 DOM 来描述 UI 的结构。虚拟 DOM 是 ReactElement 的集合,表示应用的当前状态。
6. ReactDOM.render():
- ReactDOM.render() 是 React 的核心方法之一。它的作用是将虚拟 DOM 渲染为真实 DOM。
- 具体过程:
- 接收虚拟 DOM 作为参数。
- 比较新旧虚拟 DOM,找出差异。
- 将差异应用到真实 DOM 上,更新界面。
- 这个过程通过高效的差异算法(Diffing Algorithm)实现,确保只更新必要的部分,从而提高性能。
7. 真实 DOM:
- 最终,经过 ReactDOM.render() 的处理,虚拟 DOM 的变化被应用到真实 DOM,用户界面得到更新。
React源码解析(1): JSX语法与react项目渲染过程的更多相关文章
- React源码解析之React.Children.map()(五)
一,React.Children是什么? 是为了处理this.props.children(this.props.children表示所有组件的子节点)这个属性提供的工具,是顶层的api之一 二,Re ...
- React源码解析:ReactElement
ReactElement算是React源码中比较简单的部分了,直接看源码: var ReactElement = function(type, key, ref, self, source, owne ...
- React源码解析-Virtual DOM解析
前言:最近一直在研究React,看了陈屹先生所著的深入React技术栈,以及自己使用了这么长时间.对React应该说有比较深的理解了,正好前阵子也把两本关于前端设计模式的书看完了,总感觉有一种知识错综 ...
- React源码解析——ReactAPI
一.API背景 api的具体转化关系 可以通过到https://babeljs.io/repl/网站去将我们创建的Jsx进行实时的转译 const React = { Children: { map, ...
- React源码解析:setState
先来几个例子热热身: ......... constructor(props){ super(props); this.state = { index: 0 } } componentDidMount ...
- React源码解析——创建更新过程
一.ReactDOM.render 创建ReactRoot,并且根据情况调用root.legacy_renderSubtreeIntoContainer或者root.render,前者是遗留的 API ...
- Android源码解析系列
转载请标明出处:一片枫叶的专栏 知乎上看了一篇非常不错的博文:有没有必要阅读Android源码 看完之后痛定思过,平时所学往往是知其然然不知其所以然,所以为了更好的深入Android体系,决定学习an ...
- React躬行记(16)——React源码分析
React可大致分为三部分:Core.Reconciler和Renderer,在阅读源码之前,首先需要搭建测试环境,为了方便起见,本文直接采用了网友搭建好的环境,React版本是16.8.6,与最新版 ...
- 【原】Android热更新开源项目Tinker源码解析系列之三:so热更新
本系列将从以下三个方面对Tinker进行源码解析: Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Android热更新开源项目Tinker源码解析系列之二:资源文件热更新 A ...
- 【原】Android热更新开源项目Tinker源码解析系列之一:Dex热更新
[原]Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Tinker是微信的第一个开源项目,主要用于安卓应用bug的热修复和功能的迭代. Tinker github地址:http ...
随机推荐
- 2个月搞定计算机二级C语言——真题(6)解析
1. 前言 本篇我们讲解2个月搞定计算机二级C语言--真题 6 2. 程序填空题 2.1 题目要求 2.2 提供的代码 #include <stdio.h> unsigned long f ...
- Power BI使用技巧
一.筛选器相关 1.筛选器的可选择范围不正确 在Power BI中,维度表和事实表之间的数据建模非常重要,我们在进行关系的建立的时候,PBI Desktop会根据我们的数据情况自动选择两个表之间的基数 ...
- Discuz7.2 XML漏洞
Discuz7.2 导入插件数据可以生成一句话木马,直接getshell. 姿势:版主的"管理中心" => "插件" => "导入&quo ...
- OpenAI使用AI编程给出了数数问题的解决方案 —— 如何解决ChatGPT不会数数的问题
总所周知的一个问题,那就是ChatGPT不会数数,不过今天突然发现OpenAI给出了一个神奇的解决方法,那就是AI编程. 问题案例如下: The text provided will be analy ...
- Windows安装Mysql后一段时间后Mysql服务无法启动的问题
本人在windows重装电脑后遇到一个比较麻烦的问题一直没有解决,今日有幸看到某大佬的博客得以解决.真实万分感激,特来分享一下. 第一次安装Mysql8.0之后,此次安装是将整个mysql包进行安装, ...
- Python如何根据给定模型计算权值
在机器学习和深度学习中,模型的权值(或参数)通常是通过训练过程(如梯度下降)来学习和调整的.然而,如果我们想根据一个已经训练好的模型来计算或提取其权值,Python 提供了许多工具和库,其中最常用的是 ...
- 2023NOIP A层联测22 T2 差后队列
2023NOIP A层联测22 差后队列 挺有意思的期望题,题解做法应该是 DP,但是我又双叒写出奇怪的做法-- 思路 除去最大值外的元素个数的倒数就是这一轮取到某个数的概率,而最大值是特殊的情况,在 ...
- 2.TP6的入门-分页类的改写
看了看推荐的分页类的使用,还是很简单的,可是自己去尝试改写生成的分页类结构就会很麻烦,总是不成功,后来发现手册里面还有这个 就说你想重写分页类,就需要这样做 赶紧实践了一下,先改这里的provider ...
- Java实时多任务调度过程中的安全监控设计
方浩波 (fanghb@eastcom.com)东方通信网络研究所 简介: 在一系列关联的多任务的实时环境中,如果有一个任务发生失败,可能导致所有任务产生连锁反应,从而造成调度失控的局面.特别是对于核 ...
- 在table中,tbody没有充满整个table
解决方法就是给table加上 display:table;就好了