JSX既不是字符串,也不是HTML,而是一种类似XML,用于描述用户界面的JavaScript扩展语法,如下代码所示。在使用JSX时,为了避免自动插入分号时出现问题,推荐在其最外层用圆括号包裹,并且必须用一个元素包裹(例如下面的元素)其它元素或文本,所有的元素还必须得闭合。
~~~html
(<div>
<input type="text" text={getName()} />
<button className="btn">搜索</button>
</div>)
~~~
  JSX为视图和数据架起了一座沟通的桥梁,它看起来与模板语言类似,但没有创造新的模板语法,因为JSX最终会被编译成普通的JavaScript对象,所以能够直接使用JavaScript语法。
## 一、元素
  JSX中的元素称为React元素,分为两种类型:DOM元素和组件元素,前者对应原生的HTML元素,标签的首字母要小写;后者对应自定义元素,标签的首字母要大写,如下所示。
~~~html
<button>提交</button>; //DOM元素
<Btn>自定义按钮</Btn>; //组件元素
~~~
**1)React.createElement()**
  无论是DOM元素还是组件元素,最终都会通过Babel编译器将它们转换成React.createElement()方法的调用,例如下面的元素。
~~~js
<button className="btn">搜索</button>
//编译成
React.createElement(button, { className: "btn" }, "搜索");
~~~
  React.createElement()能接收3个参数(如下所示),其中type是元素类型,也就是它的名称;props是一个由元素属性组成的对象;children是它的子元素(即内容),可以是文本也可以是其它元素。
~~~js
React.createElement(type, [props], [...children])
~~~
  方法的返回值是一个元素对象,简化过的对象如下所示。
~~~json
{
type: "button",
props: {
className: "btn",
children: "搜索"
}
}
~~~
  为了避免在多人协作时出现相同名称的元素,可以为元素添加命名空间,例如调用UI模块中的Btn元素,可以像下面这么写。
~~~js
const UI = {
Btn: function(props) {
return <button className={props.className}>{props.children}</button>;
}
}
<UI.Btn className="btn">搜索</UI.Btn>
~~~
**2)注释**
  JSX中的注释需要像下面这样,用一对花括号、斜杠和星号包裹。
~~~js
{/* 表单中的提交按钮 */}
<button>提交</button>
~~~
**3)表达式**
  在JSX的任意位置都能插入表达式,但必须用花括号包裹住才能有效,例如像下面这样调用getName()函数。注意,在JSX中不能插入语句。
~~~js
function getName() {
return "strick";
}
<div>{getName()}</div>
~~~
  由于JSX本身就是一种表达式,因此它可以作为函数的参数、返回值或变量的值,如下所示。
~~~js
if (true) {
let fragment = <div>{getName()}</div>;
}
~~~
  在JSX中传入的值都会自动被HTML转义,这样可以防止XSS攻击,例如输入“\<p>\</p>”,输出“\<p\>\</p\>”,如下所示。
~~~js
//<p></p>
<div>{"<p></p>"}</div>
~~~
  如果要输出不转义的值,那么可以用React提供的dangerouslySetInnerHTML属性,如下代码所示。它的值是一个包含\_\_html属性的对象,其作用相当于调用DOM元素的innerHTML属性。
~~~html
<div dangerouslySetInnerHTML={{__html: "<p></p>"}}></div>
~~~
**4)内容**
  当元素的内容是字符串时,JSX会移除字符串中的空行,其内部的换行会被替换成一个空格,下面的两个元素是等价的。
~~~html
<p>freedom strick</p>
<p>
freedom
strick
</p>
~~~
  当元素的内容是false、null、undefined或true时,它们都不会被渲染到DOM结构中,因此下面的五个元素是等价的。
~~~html
<p></p>
<p>{false}</p>
<p>{null}</p>
<p>{undefined}</p>
<p>{true}</p>
~~~
**5)渲染**
  如果要将React元素渲染到页面的DOM结构中,可以调用ReactDOM.render()方法,此方法接收3个参数,如下所示。
~~~js
ReactDOM.render(element, container[, callback])
~~~
  element是要渲染的元素;container是页面中的一个节点,在此处起到容器的作用,element会被渲染到container中;callback是可选的回调函数,会在组件被渲染或更新之后触发。此方法的使用可参考下面的示例。
~~~html
<div id="container">freedom</div>
<script type="text/babel">
ReactDOM.render(
<p>strick</p>,
document.getElementById("container")
);
</script>
~~~
  当第一次调用ReactDOM.render()方法时,容器内部的元素会被全部替换掉,也就是执行上面的代码得到的结果如下所示,原先的字符串“freedom”被替换成了元素。
~~~html
<div id="container">
<p>strick</p>
</div>
~~~
## 二、属性
  React对元素属性进行了一次封装,不仅规范了属性的命名,还完善了浏览器的兼容性。在JSX中,DOM元素的属性对应标准的DOM属性和特性;而组件元素的属性都是无对应关系的自定义属性。除了以“data-”和“aria-”为前缀的元素属性要用小写命名之外,其余的都得遵循小驼峰命名法,例如maxlength变成maxLength、onclick变成onClick等。还有两个比较特殊的属性:class和for,由于它们是JavaScript的关键字,因此需要变成className和htmlFor后才能使用。
**1)默认值**
  属性的默认值是true,下面的两个元素是等价的,页面上的显示如图1所示。
~~~html
<input type="text" value />
<input type="text" value={true} />
~~~
:-: ![](https://box.kancloud.cn/5c5af1ac515467a782c520e3a51d939a_341x121.png)
:-: 图 1 带默认值的文本框
  在标准的DOM中,诸如checked、disabled等布尔属性,它们的值要么为空要么为对应的关键字,例如“checked”、“disabled”;而JSX中的布尔属性,它们的值只能是true或false。
**2)字符串和表达式**
  当属性的值是字符串时,其值需要用双引号包裹;当属性的值是表达式时,其值需要用花括号包裹,如下所示。
~~~html
<input type="text" value="3" />
<input type="text" value={1 + 2} />
~~~
**3)扩展属性**
  如果存在一个由元素属性组成的属性对象,那么就能利用ES6新增的扩展运算符,把属性对象展开并传递给元素,如下所示。
~~~js
var props = { type: "text", value: "1" };
<input {...props} />
//相当于
<input type="text" value="1" />
~~~
  相比直接在元素上设置属性,这种方式操作起来更加灵活。
## 三、虚拟DOM
  HTML文档能被抽象成一棵由多种类型的节点构成的DOM树,而每次对DOM节点执行增删改查等操作,往往会触发非常消耗性能的重绘和重排。为了解决这个性能瓶颈,React引入了虚拟DOM。虚拟DOM(Virtual DOM)是构建在真实DOM之上的一层抽象,它将DOM元素映射成内存中的JavaScript对象(即通过React.createElement()得到的React元素),形成一棵JavaScript对象树。
  在React中,将虚拟DOM转换成真实DOM的过程叫做调和(Reconciliation),而diff算法是保证调和高效的关键,因为diff算法会找出新旧虚拟DOM之间的差异部分,随后只更新真实DOM中需要变化的节点,而不是将整棵DOM树重新渲染一遍。经过虚拟DOM的隔离,开发人员已经不用再直接与页面上的真实DOM打交道了,如图2所示。
:-: ![](https://box.kancloud.cn/9535161bb5b2e47fb1336d620885e99c_1234x318.png =600x)
:-: 图 2 新的开发模式
  虚拟DOM还有一大亮点,那就是将它与其他渲染器配合能够集成到指定的终端,即将React元素映射成对应的原生控件,前文所描述的是用react-dom在Web端渲染,还可以使用react-native在手机端(Android或iOS)渲染。
*****
> 原文出处:
[博客园-React躬行记](https://www.cnblogs.com/strick/category/1455720.html)
[知乎专栏-React躬行记](https://zhuanlan.zhihu.com/pwreact)
已建立一个微信前端交流群,如要进群,请先加微信号freedom20180706或扫描下面的二维码,请求中需注明“看云加群”,在通过请求后就会把你拉进来。还搜集整理了一套[面试资料](https://github.com/pwstrick/daily),欢迎浏览。
![](https://box.kancloud.cn/2e1f8ecf9512ecdd2fcaae8250e7d48a_430x430.jpg =200x200)
推荐一款前端监控脚本:[shin-monitor](https://github.com/pwstrick/shin-monitor),不仅能监控前端的错误、通信、打印等行为,还能计算各类性能参数,包括 FMP、LCP、FP 等。
- ES6
- 1、let和const
- 2、扩展运算符和剩余参数
- 3、解构
- 4、模板字面量
- 5、对象字面量的扩展
- 6、Symbol
- 7、代码模块化
- 8、数字
- 9、字符串
- 10、正则表达式
- 11、对象
- 12、数组
- 13、类型化数组
- 14、函数
- 15、箭头函数和尾调用优化
- 16、Set
- 17、Map
- 18、迭代器
- 19、生成器
- 20、类
- 21、类的继承
- 22、Promise
- 23、Promise的静态方法和应用
- 24、代理和反射
- HTML
- 1、SVG
- 2、WebRTC基础实践
- 3、WebRTC视频通话
- 4、Web音视频基础
- CSS进阶
- 1、CSS基础拾遗
- 2、伪类和伪元素
- 3、CSS属性拾遗
- 4、浮动形状
- 5、渐变
- 6、滤镜
- 7、合成
- 8、裁剪和遮罩
- 9、网格布局
- 10、CSS方法论
- 11、管理后台响应式改造
- React
- 1、函数式编程
- 2、JSX
- 3、组件
- 4、生命周期
- 5、React和DOM
- 6、事件
- 7、表单
- 8、样式
- 9、组件通信
- 10、高阶组件
- 11、Redux基础
- 12、Redux中间件
- 13、React Router
- 14、测试框架
- 15、React Hooks
- 16、React源码分析
- 利器
- 1、npm
- 2、Babel
- 3、webpack基础
- 4、webpack进阶
- 5、Git
- 6、Fiddler
- 7、自制脚手架
- 8、VSCode插件研发
- 9、WebView中的页面调试方法
- Vue.js
- 1、数据绑定
- 2、指令
- 3、样式和表单
- 4、组件
- 5、组件通信
- 6、内容分发
- 7、渲染函数和JSX
- 8、Vue Router
- 9、Vuex
- TypeScript
- 1、数据类型
- 2、接口
- 3、类
- 4、泛型
- 5、类型兼容性
- 6、高级类型
- 7、命名空间
- 8、装饰器
- Node.js
- 1、Buffer、流和EventEmitter
- 2、文件系统和网络
- 3、命令行工具
- 4、自建前端监控系统
- 5、定时任务的调试
- 6、自制短链系统
- 7、定时任务的进化史
- 8、通用接口
- 9、微前端实践
- 10、接口日志查询
- 11、E2E测试
- 12、BFF
- 13、MySQL归档
- 14、压力测试
- 15、活动规则引擎
- 16、活动配置化
- 17、UmiJS版本升级
- 18、半吊子的可视化搭建系统
- 19、KOA源码分析(上)
- 20、KOA源码分析(下)
- 21、花10分钟入门Node.js
- 22、Node环境升级日志
- 23、Worker threads
- 24、低代码
- 25、Web自动化测试
- 26、接口拦截和页面回放实验
- 27、接口管理
- 28、Cypress自动化测试实践
- 29、基于Electron的开播助手
- Node.js精进
- 1、模块化
- 2、异步编程
- 3、流
- 4、事件触发器
- 5、HTTP
- 6、文件
- 7、日志
- 8、错误处理
- 9、性能监控(上)
- 10、性能监控(下)
- 11、Socket.IO
- 12、ElasticSearch
- 监控系统
- 1、SDK
- 2、存储和分析
- 3、性能监控
- 4、内存泄漏
- 5、小程序
- 6、较长的白屏时间
- 7、页面奔溃
- 8、shin-monitor源码分析
- 前端性能精进
- 1、优化方法论之测量
- 2、优化方法论之分析
- 3、浏览器之图像
- 4、浏览器之呈现
- 5、浏览器之JavaScript
- 6、网络
- 7、构建
- 前端体验优化
- 1、概述
- 2、基建
- 3、后端
- 4、数据
- 5、后台
- Web优化
- 1、CSS优化
- 2、JavaScript优化
- 3、图像和网络
- 4、用户体验和工具
- 5、网站优化
- 6、优化闭环实践
- 数据结构与算法
- 1、链表
- 2、栈、队列、散列表和位运算
- 3、二叉树
- 4、二分查找
- 5、回溯算法
- 6、贪心算法
- 7、分治算法
- 8、动态规划
- 程序员之路
- 大学
- 2011年
- 2012年
- 2013年
- 2014年
- 项目反思
- 前端基础学习分享
- 2015年
- 再一次项目反思
- 然并卵
- PC网站CSS分享
- 2016年
- 制造自己的榫卯
- PrimusUI
- 2017年
- 工匠精神
- 2018年
- 2019年
- 前端学习之路分享
- 2020年
- 2021年
- 2022年
- 2023年
- 日志
- 2020