生命周期钩子
关于生命周期钩子,官方文档给出了详细的说明
React生命周期 - 官方文档
不同阶段调用的生命周期钩子
挂载阶段
当组件实例被创建并插入 DOM 中时,其生命周期调用顺序如下:
constructor()
static getDerivedStateFromProps()
render()
componentDidMount()
更新阶段
当组件的 props 或 state 发生变化时会触发更新。组件更新的生命周期调用顺序如下:
static getDerivedStateFromProps()
shouldComponentUpdate()
render()
getSnapshotBeforeUpdate()
componentDidUpdate()
卸载阶段
当组件从 DOM 中移除时会调用如下方法:
componentWillUnmount()
生命周期钩子函数介绍
constructor(props)
构造函数,创建实例时候会调用,通常用于绑定函数,初始化state。
static getDerivedStateFromProps(props, state)
静态方法,用于state的某些值只依赖于当前的props的场景,并且在初始挂载及后续更新时都会被调用
render()
渲染函数,返回JSX或者React.element。
componentDidMount()
挂载完成,通常在这个钩子中做初始化操作,请求接口、绑定事件、启动定时器等。
shouldComponentUpdate()
用于优化性能,判断是否需要比较更新。
getSnapshotBeforeUpdate(prevProps, prevState)
它使得组件能在发生更改之前从 DOM 中捕获一些信息(例如,滚动位置)。此生命周期的任何返回值将作为参数传递给 componentDidUpdate()。
componentDidUpdate(prevProps, prevState, snapshot)
更新完成,做一些有副作用的操作,例如请求接口,弹出提示等等。
prevProps和prevState是更新前的props和state,用于逻辑判断中需要对比更新的改动。
componentWillUnmount()
会在组件卸载及销毁之前直接调用。在此方法中执行必要的清理操作,如解除订阅、清除定时器。
getDerivedStateFromProps和getSnapshotBeforeUpdate示例
【前端面试刷题网站:灵题库,收集大厂面试真题,相关知识点详细解析。】
对于getDerivedStateFromProps和getSnapshotBeforeUpdate不是很容易理解,给出示例代码:
getDerivedStateFromProps对props传入的digit和Child的state的sum计算求和。
注意getDerivedStateFromProps是静态方法,因此没有this。getDerivedStateFromProps的作用就是根据props和state计算新的state,不需要this。
import React from 'react';
class Child extends React.PureComponent {
// state完全依赖于props.digit,因此可以使用getDerivedStateFromProps来进行处理
static getDerivedStateFromProps(props, state) {
return {
sum: state.sum + props.digit
};
}
state = {
sum: 0
};
render() {
return (
累计:{this.state.sum}
);
}
}
class App extends React.PureComponent {
state = {
digit: 0
};
onButtonClick = () => {
this.setState(preState => ({
digit: preState.digit + 1
}));
};
render() {
return (
数字:{this.state.digit}
);
}
}
export default App;
对getSnapshotBeforeUpdate的官方文档的示例代码稍加改造。
App有个按钮元素,点击则给list增加元素,ScrollingList组件渲染list,并且如果列表处于底部时候,增加新元素后会自动滚动到底部。
import React from 'react';
import './App.css';
class ScrollingList extends React.Component {
constructor(props) {
super(props);
this.listRef = React.createRef();
}
getSnapshotBeforeUpdate(prevProps, prevState) {
// 如果有增加的item,则可能需要滚动
if (prevProps.list.length < this.props.list.length) {
const list = this.listRef.current;
// 如果当前没有在回看,需要滚动
return list.scrollHeight - list.scrollTop - list.clientHeight === 0;
}
return false;
}
componentDidUpdate(prevProps, prevState, snapshot) {
// 如果我们 snapshot 有值,说明我们刚刚添加了新的 items,
// 并且没有在会看,说明我们需要滚动到底部
//(这里的 snapshot 是 getSnapshotBeforeUpdate 的返回值)
if (snapshot) {
const list = this.listRef.current;
list.scrollTop = list.scrollHeight;
}
}
render() {
return (
{
this.props.list.map(text => (
{text}
))
}
);
}
}
class App extends React.PureComponent {
state = {
list: []
};
onButtonClick = () => {
this.setState(preState => ({
list: [...preState.list, preState.list.length]
}));
};
render() {
return (
);
}
}
export default App;
React16更新的生命周期钩子
移除的
UNSAFE_componentWillMount()
UNSAFE_componentWillReceiveProps(nextProps)
UNSAFE_componentWillUpdate(nextProps, nextState)
新增的
static getDerivedStateFromProps(props, state)
getSnapshotBeforeUpdate(prevProps, prevState)
关于setState
setState是异步的吗?
出于性能考虑,React将setState进行异步处理,即调用setState时候不会马上更新,而是将setState的值放入一个队列,然后延时批量处理队列中的state。在生命周期钩子和React合成事件中中调用setState,会将数据添加到一个队列中,合并处理,这时候setState是异步的。而在其他的React控制之外的地方调用(比如原生dom事件)则不是异步的。
setState异步特性,导致需要注意两个问题
- 设置完state,获取数据的时机
由于异步特性,setState改变state数据后,无法马上获取到state的最新值。
由于setState是异步的,如果想获取改变后的值,应该在setState的第二个参数回调函数中访问state的最新值。
- 修改数据依赖现有的数据
由于setState会异步合并处理,因此如果设置的state值依赖于之前的state值,需要给setState传一个函数,
函数参数是之前的state,返回的值是新的改变的state属性。
import React from 'react';
class App extends React.Component {
state = {
digit: 1,
text: 'a'
};
componentDidMount() {
document.getElementById('btn')
.addEventListener('click', this.onAddButtonClick);
}
onAddButtonClick = () => {
// bad
// const add1 = () => {
// this.setState({digit: this.state.digit + 1});
// };
// add1();
// add1();
// add1();
// good
const add1 = () => {
this.setState((state) => {
return {digit: state.digit + 1};
});
};
add1();
add1();
add1();
};
onAddButtonClick2 = () => {
// work
const add1 = () => {
this.setState({digit: this.state.digit + 1});
};
add1();
add1();
add1();
};
onChangeButtonClick = () => {
// bad
// this.setState({text: 'b'});
// console.log(this.state.text);
// good
this.setState({text: 'b'}, () => {
console.log(this.state.text);
});
};
render() {
return (
{this.state.digit}
{this.state.text}
);
}
}
export default App;
React更新界面的主要过程可以简单描述为,调用setState之后,React会更新state,然后调用组件的render得到新的state对应的虚拟dom,然后对比当前的和更新后端虚拟dom,判断是否需要更新dom,以及如何更新dom。
import React from 'react';
import './App.css';
class App extends React.Component {
state = {
digit: 1
};
componentDidMount() {
// 即使digit没有变,也还是会触发对比,调用render方法
this.setState({digit: 1});
}
render() {
console.log('render');
return (
{this.state.digit}
);
}
}
export default App;
由于diff工作量比较大,如果setState的值没有改变,其实是不需要进行diff的。如何让state没有改变的时候,不进行diff呢?可以使用shouldComponentUpdate这个生命周期钩子。
在调用setState更新数据后,React会判断是否需要进行更新操作,如果没有shouldComponentUpdate这个生命周期钩子,则默认进行对比和更新工作,如果有shouldComponentUpdate,则调用之,如果返回true则更新,返回false则不更新。利用这个钩子,我们可以对新的state和当前的state进行对比,如果有变化,返回true,如果没有变化,返回false。
React.PureComponent实现了浅比较的shouldComponentUpdate,因此我们的组件如果继承了 React.PureComponent就会有了对比state决定是否更新的特性。
import React from 'react';
import './App.css';
class App extends React.PureComponent {
state = {
digit: 1
};
componentDidMount() {
// 继承PureComponent,当数据未改变时候,不会触发对比更新
this.setState({digit: 1});
}
render() {
console.log('render');
return (
{this.state.digit}
);
}
}
export default App;
但是使用React.PureComponent时候需要注意一个问题,浅比较是比较值是否相同,因此当state中的数据是一个对象,其中属性变化但引用不变、或者state中的数据是一个数组,数组中的元素变化但数组引用不变,这时候比较结果是两者相同,因此不会触发更新。如果希望在对象属性变化、数组元素变化时候触发更新,应该setState时候传入新的对象或数组。
import React from 'react';
import './App.css';
class App extends React.PureComponent {
state = {
obj: {
text: 'obj'
},
arr: [1, 2]
};
componentDidMount() {
this.state.obj.text = 'obj1';
this.state.arr.push(3);
// obj和arr引用未变,因此不会触发更新
// this.setState({
// obj: this.state.obj,
// arr: this.state.arr
// });
// obj和arr复制为新的对象/数组,因此会触发更新
this.setState({
obj: {...this.state.obj},
arr: [...this.state.arr]
});
}
render() {
console.log('render');
return (
{this.state.obj.text}
{this.state.arr.join(',')}
);
}
}
export default App;
通常情况下,我们应该让组件继承React.PureComponent,并且对象和数组变化时候要传入新的引用。这样既可以利用shouldComponentUpdate的浅比较避免不必要的对比更新操作,也能保证在需要的时候更新视图。
虚拟dom
了解虚拟dom有助于理解React执行原理
我们已经知道,React框架会帮助我们进行dom操作,我们只需要实现组件,当组件的状态发生变化时候,框架需要更新视图,最简单的方法是重新构建整个视图,但是这样做会产生昂贵的性能损耗。实际上,每次状态变化时候,React会根据render结果更新虚拟dom树,然后对比当前虚拟dom和更新后的虚拟dom,然后以最小的代价更新实际dom。
React的diff算法依据两个假设
① 两个不同类型的元素会产生出不同的树
② 开发者可以通过 key prop 来暗示哪些子元素在不同的渲染下能保持稳定
因此,在进行diff时候,如果两个节点是不同类型的dom(比如一个是div,一个是span)、或者是不同的组件(比如一个是Icon,一个是Modal)、或者两个节点的key值不同,则认为是不同类型节点,否则认为是相同的节点。
对比不同类型节点时候,直接卸载调之前的节点及其所有子节点,重新建立新的节点
对比相同类型节点时候,如果是原生的元素(如div、img),对比并更新有变化的属性;如果是组件,更新组件的props。
对比子节点时候,按顺序对比每个子节点,即当前的第一个子节点和新的第一个子节点对比、当前的第二个子节点和新的第二个子节点对比......,如果是同类型的节点,按照上述同类型的规则处理,如果是不同类型则按照上述不同类型节点的处理。
处理一个列表组件时候,可能存在性能问题,见官方示例
- Duke
- Villanova
- Connecticut
- Duke
- Villanova
在列表头部添加一个元素,由于React是按顺序对比,因此认为3个元素都有变化,这时候React会把三个元素都进行修改。可以通过添加key属性来让React更好地识别哪些元素改动。
- Duke
- Villanova
- Connecticut
- Duke
- Villanova
添加key之后,React会知道新的列表只是在头部添加了一个元素,其他两个元素因为key能够对应,因此没有变化,所以React会只操作添加的新dom。
函数式组件
一个返回JSX的函数,就是一个函数式组件。
如果一个组件没有生命周期钩子和state,可以写成函数式组件的形式。
函数式组件更简洁,相比于class组件省去了创建实例的损耗,但是每次数据改变都需要进行对比工作,不能像class组件一样使用shouldComponentUpdate避免不必要的对比更新。
有了hook API之后,函数式组件也可以维护状态并在特定的声明周期做一些处理了。
export default () => (
functional component
);
组件复用
在React中有很多组件复用的方式,包括HOC,render props、hook,它们在不同的场景有各自作用。
1. 高阶组件(HOC)
高阶组件(HOC)是 React 中用于复用组件逻辑的一种高级技巧。HOC 自身不是 React API 的一部分,它是一种基于 React 的组合特性而形成的设计模式。
具体而言,高阶组件是参数为组件,返回值为新组件的函数。
组件是将 props 转换为 UI,而高阶组件是将组件转换为另一个组件。
根据官网定义,高阶组件是一种复用组件的技巧。
关键点:高阶组件是一个函数,输入是一个组件(输入可能还会包含其他的参数,其他参数用来自定义组件的特性),输出是一个组件。
当我们拥有一个组件,而且需要在不同情况下对组件进行改造生成新的组件时候,高阶组件会有作用。
import React from 'react';
// withList是一个高阶组件,它包含了list的逻辑,给WrappedComponent组件传入list
// length参数可以自定义list的长度
function withList(WrappedComponent, length) {
return class List extends React.PureComponent {
state = {
list: (new Array(length)).fill(123)
};
render() {
return
}
}
}
class App extends React.PureComponent {
constructor(props) {
super(props);
this.state = {
skin: 'red'
};
}
onButtonClick = () => {
this.setState(prevState => ({
skin: prevState.skin === 'red' ? 'blue' : 'red'
}));
};
render() {
return (
{this.props.list.map((text, index) => (
- {text}
))}
);
}
}
export default withList(App, 10);
2. render props
术语 “render prop” 是指一种在 React 组件之间使用一个值为函数的 prop 共享代码的简单技术
render props也不是React的API,而是一种复用组件的技术。
关键点:给组件传入一个方法props,组件在render中调用这个方法来渲染界面。
当我们拥有一个组件,它维护了可复用的数据和逻辑,我们需要使用组件的数据和逻辑,并且希望自定义不同的界面时候,会用到render props。
见下面示例
List组件维护list的逻辑,但它不负责界面,界面渲染交给通过props传入的render方法。
App组件使用了List,并传入Skin作为render,Skin负责界面渲染,而Skin渲染的数据list来自List组件。
这样如果还有其他的组件使用List渲染不同的界面,都可以复用List组件。
import React from 'react';
class List extends React.PureComponent {
constructor(props) {
super(props);
this.state = {
list: (new Array(props.length)).fill(123)
};
}
render() {
return (
{this.props.render(this.state.list)}
);
}
}
class Skin extends React.PureComponent {
constructor(props) {
super(props);
this.state = {
skin: 'red'
};
}
onButtonClick = () => {
this.setState(prevState => ({
skin: prevState.skin === 'red' ? 'blue' : 'red'
}));
};
render() {
return (
{this.props.list.map((text, index) => (
- {text}
))}
);
}
}
class App extends React.PureComponent {
render() {
return } />
}
}
export default App;
3. hook
简介
什么是hook?
Hook 是一个特殊的函数,它可以让你“钩入” React 的特性
这里的React特性指的是生命周期、state状态等。
当我们希望使用函数式组件,并且希望能够让不同功能代码更聚合,更好维护时候,可以使用hook。
为什么需要hook?
为了让功能代码聚合在一起,增加了组件的可维护性。hook提供了功能逻辑代码复用的新的方式。React需要提供一些额外的API来实现,由于React更推崇函数式组件的简洁和声明式编程的理念,因此将这种API增加在了函数式组件中,而非class组件中。
hook给函数式组件增加状态和逻辑,并将功能代码聚合在一起。
hook的规则:
只能在函数最外层调用 Hook。不要在循环、条件判断或者子函数中调用。
只能在 React 的函数组件中调用 Hook。不要在其他 JavaScript 函数中调用。(还有一个地方可以调用 Hook —— 就是自定义的 Hook 中)
自定义hook
自定义hook是一个使用了React的原生hook和其他自定义hook的函数。自定义的hook就是一个功能复用的单元。
常用的hook
useState和useEffect
在函数式组件中添加状态需要使用useState这个hook;在函数式组件中添加一些有副作用的操作(如网络请求、操作DOM、绑定事件回调),需要用到useEffect这个hook。
useState
- useState函数接收一个参数作为初始值
- useState返回值为一个数组,数组第一个元素是状态变量,第二个元素是设置该状态的方法
- 调用设置状态的方法,会同步触发diff和更新操作
注意:相比单个state,更推荐多个state,因为单个state无法达到聚合功能代码的目的,所以应该每个state属性都调用useState创建一个对应的状态。我应该使用单个还是多个 state 变量?
useEffect
- useEffect接收第一个参数是一个回调callback,会在特定时机执行
- 给useEffect传入的callback,可以返回一个方法clearEffect,React会在组件卸载时候调用clearEffect,清除副作用(比如停止定时器,解除事件绑定)
- useEffect第二个参数是一个数组,数组中是依赖项,当依赖变化时候才会执行useEffect传入的方法
- 如果是空数组,则只在组件初始挂载时候执行,类似componentDidMount
- 如果第二个参数不传,则在组件更新时候执行,类似componentDidMount和componentDidUpdate
下面看一个简单的示例
useState返回的set方法,当数值相同时候不会触发子组件render,这个和class组件不同。另外每次set都会立即触发更新,而不是像this.setState一样,合并异步更新。
import React, {useState, useEffect} from 'react';
const ChildComponent = props => {
let [digit, setDigit] = useState(0);
useEffect(
() => {
const timer = setInterval(
() => {
// 立即触发更新,不会有延时
setDigit(digit++);
},
1000
);
return () => {
clearInterval(timer);
}
},
[]
);
return {digit};
}
class App extends React.PureComponent {
state = {
showChild: true
};
componentDidMount() {
setTimeout(
() => {
this.setState({showChild: false});
},
10 * 1000
);
}
render() {
return this.state.showChild &&
}
};
export default App;
useMemo和useCallback
useMemo的作用类似于Vue中的computed,会缓存计算结果,只有当依赖的数据改变时候,才会执行其中的计算方法。
useCallback一般用于生成事件处理函数,防止每次生成新的函数,导致不必要的子组件重新render。
场景是计算属性和注册回调。
useMemo
- useMemo接收的第一个参数是一个函数,这个函数用来根据state计算新的值。
- useMemo接收的第二个参数是一个依赖项数组。如果不传,则每次都会调用计算函数,没有任何优化效果;如果传空数组,则只在第一次调用计算函数,返回的结果永远不变;如果数组传入计算的依赖项,则依赖项变化时候调用计算函数,得到最新的结果,在依赖项不变时候,返回之前的结果,因此起到了“缓存计算结果”的效果。
useCallback
- useCallback接收第一个参数callback。
- useCallback接收第二个参数,是依赖项。
- useCallback返回一个函数,如果依赖项不传,则每次生成一个新的函数;如果依赖项传空数组,则只在第一次生成一个函数,后面每次render都使用这个函数;如果传入依赖项,则当依赖项发生变化时候生成一个新的函数。
- 注意,useCallback的回调中如果使用了某个state,就会形成一个闭包,如果给useCallback传入第二个参数是空数组,则生成的函数引用的永远是state初始化的值,因此如果回调中依赖了某个state,应该把依赖项加到第二个参数的数组中。(看下面注释加深理解)
useCallback(fn, deps)
相当于 useMemo(() => fn, deps)
。
看下面简单的示例
import React, {useState, useMemo, useCallback} from 'react';
class Button extends React.PureComponent {
render() {
console.log('render Button');
return
}
}
const App = () => {
console.log('render App');
const [digit, setDigit] = useState(0);
// 避免了Button组件每次点击计算按钮时候都会更新
const onButtonClick = useCallback(() => {
setDigit(document.getElementById('_input').value);
// 这里访问的digit是在调用App方法时候,useState返回的
// 每次调用App都会返回一个新的变量(但值可能不会变)
// 如果第二个参数是"[]",则onButtonClick一直不会变
// 初始的回调中访问的digit是0,因此后面即使digit改变,调用onButtonClick还是打印0
// 除非在依赖项数组中加入digit,onButtonClick在digit改变时候重新创建新的函数,就可以打印最新的digit了
console.log(digit);
}, []);
// 当digit不变时,square不会重新计算,而是返回之前的值
const square = useMemo(() => {
console.log('calculate square')
return digit * digit;
}, [digit]);
return (
);
};
export default App;
useRef
useRef可以用来实现获取一个原生DOM或者子组件实例的引用,本质上,它可以用来保存任何可变值。
- useRef接收一个参数作为初始值。
- useRef返回一个对象,这个对象包含一个current属性,这个current属性会被初始化为初始值,并且可以更改。
- 每次调用渲染函数,都返回同一个对象。
看下面的示例
// 获取input元素的引用
function TextInputWithFocusButton() {
const inputEl = useRef(null);
const onButtonClick = () => {
// `current` 指向已挂载到 DOM 上的文本输入元素
inputEl.current.focus();
};
return (
<>
>
);
}
// 用来保存timer的id
function Timer() {
const intervalRef = useRef();
useEffect(() => {
intervalRef.current = = setInterval(() => {
// ...
});
}, []);
const handleCancelClick = useCallback(() => {
clearInterval(intervalRef.current);
}, []);
// ...
}
useState原理
useState让函数式组件能够拥有自己的状态,如何让函数式组件拥有自己的状态呢?有两个关键:
- 状态存储在哪里?
- 状态以什么格式存储?
状态存储在哪里呢?对于class组件可以直接挂载实例上面,函数式组件又应该存储在哪里呢?答案是挂在虚拟DOM节点上,渲染过程中,调用函数组件时候React是知道在处理哪个节点的,因此当在函数式组件中调用useState方法,React就可以把创建的状态挂到相应的节点上面。在下一次调用函数组件时候,React发现已经有了这个状态,就把值返回。
状态如何存储?首先我们想到可以类比class组件,把整个state作为一个对象存储,但是上面提到,这样存储不能达到分离功能逻辑的目的;因此我们希望可以将状态分开,每个状态有自己对应的key,例如这样:
const [a, setA] = useState('a', '');
即用对象存储状态,useState方法传递一个key和一个初始值,React把这个key和对应value存储到对象上。但是这样会比较冗余。React实现方式是只需要传递初始值,不用传递key,React根据useState调用的顺序来识别是哪一个状态。那么函数式组件的状态就是一个数组格式,每个状态在函数式组件中调用useState声明的顺序,也是它们在状态数组中存储的顺序。
这也是useState必须在函数顶层调用的原因。
useState工作的过程如下:
- 首次调用函数组件,执行useState,React获取到当前正在遍历的节点,创建_state对象挂载到节点上,根据调用useState的顺序,在数组中加入状态,并根据传入的初始值初始化状态。
- 当调用状态的set方法时候,React修改_state上面的状态,并更新相应的组件。
- 后面再调用函数组件时候,useState找到对应的节点的_state,并根据useState的调用顺序,找到修改后的状态返回,这样组件就可以拿到正确的状态了。
context
1. 什么是context
context是React提供的多层组件传递数据的解决方案,它包括多个API。
- React.createContext
- context.Provider
- context.Consumer
- class.contextType
使用context,开发者需要关注几个点。
祖先组件如何将数据传递下去
-
子组件如何接收数据
- class component如何接收数据
- functional component如何接收数据
-
如何动态修改数据
- 祖先组件如何修改数据
- 子组件如何修改数据
如何接受多个context
2. 为什么需要context
为了避免多层组件传递数据使用prop,导致传递数据的组件和使用数据的组件中间的组件需要处理不必要的数据,增加代码复杂度,影响组件复用性,我们可以使用context来满足多层组件传递数据的需求。
3. 如何使用context
1. 父组件提供context
// 默认值
const ThemeContext = React.createContext('1');
// 将theme数据传递下去
const App = () => (
);
2. 子组件接受数据
2.1 class component接受数据
class Pages extends React.Component {
componentDidMount() {
console.log(this.context);
}
}
Pages.contextType = ThemeContext;
2.2 functional component接受数据。注意,子组件接受的是最近Provider提供的值
const WrappedPages = () => (
{
theme => (
)
}
);
3. 如何动态修改数据
3.1 祖先组件如何修改数据
// 创建context
const ThemeContext = React.createContext('1');
// 使用state可以动态修改context
class App extends React.Component {
state = {
theme: '1'
};
componentDidMount() {
this.setState({
theme: '2'
});
}
render() {
return (
);
}
};
3.2 子组件如何修改数据
// 创建context
const ThemeContext = React.createContext('1');
// 将修改theme的回调传递下去
class App extends React.Component {
state = {
theme: '1'
};
changeTheme = theme => {
this.setState({theme});
};
render() {
return (
);
}
};
// 调用回调修改context
const Pages = ({changeTheme}) => (
<>
{
// 根据theme渲染一些元素
}
>
);
4. 如何消费多个context
// 嵌套函数包裹子组件,将多个context传递给子组件
const ThemeContext = React.createContext('1');
const EnvContext = React.createContext('test');
const App = () => (
{
theme => (
{
env => (
)
}
)
}
);
错误边界
1. 背景
React16对待组件渲染错误的策略是,任何未被捕获的错误将会导致整个React组件树整个被卸载。
因此React提供了一种机制让开发者捕获子组件树中的错误。这就是错误边界。
2. 错误边界相关的API
- 如果一个class组件定义了
static getDerivedStateFromError
或者componentDidCatch
中的任何一个,它就变成一个错误边界 - 当抛出错误后,使用
static DerivedStateFromError
渲染备用UI,使用componentDidCatch
打印信息。
3. 错误边界捕获的信息
错误边界捕获的是其子组件树的错误,它不能捕获
- 事件处理
- 异步代码
- 服务端渲染
- 它自身抛出的错误
【思考】由于异常捕获是try catch子组件生命周期钩子中的错误,因此上述情况无法捕获。
4. 错误边界应该放在哪里
大多数情况下, 你只需要声明一次错误边界组件, 并在整个应用中使用它。
- 放在全局,以告诉开发者具体的子组件的错误
- 可以将单独的部件(如第三方React组件)包在错误边界中,以保证不会影响到其他组件
5. 组件栈追踪
React16会把渲染期间发生的所有错误打印到控制台。
打印的信息除了错误信息和JavaScript调用堆栈外,React还支持组件栈追踪。即查看发生错误的组件名和行号。
组件栈错误追踪的功能需要相关支持,React需要知道: 1. 该组件的名称和 2. 源代码行号。这是通过一些第三方工具实现的
- 目前主流浏览器都支持function.name,这样React可以通过类的name属性知道组件的名称。对于不支持function.name的浏览器,可以通过
function.name-polyfill
或者组件上挂载displayName
属性来实现。 - 使用babel插件babel-plugin-transform-react-jsx-source
给组件添加__source
属性
6. 示例代码
// 定义错误边界
class ErrorBoundary extends React.Component {
constructor(props) {
super(props);
this.state = { hasError: false };
}
static getDerivedStateFromError(error) {
// 更新 state 使下一次渲染能够显示降级后的 UI
return { hasError: true };
}
componentDidCatch(error, info) {
// 你同样可以将错误日志上报给服务器
logErrorToMyService(error, info);
}
render() {
if (this.state.hasError) {
// 你可以自定义降级后的 UI 并渲染
return Something went wrong.
;
}
return this.props.children;
}
}
// 使用错误边界
代码分割
1. 需求背景
组件代码分割的需求:有些组件可能在初始化时候不需要加载,后面需要的时候再加载。这样就需要我们对组件分片,构建时候该组件相关代码单独打包成一个代码片,运行时候按需加载下来并执行渲染。
组件代码分割需求有以下两个要点
- 构建:如何将组件代码分片打包,并异步渲染
- loading:在组件异步加载过程中,需要展示loading,如何在合适时机展示和移除loading
2. 代码分割的实现
- 第一个需求,组件代码分片打包。使用es6的动态import实现分片打包(需要
babel-plugin-syntax-dynamic-import
的支持),使用React.lazy
实现异步加载。
上述代码会在加载OtherComponent
时候才去下载并执行渲染,如果React不渲染MyComponent
,则不会加载OtherComponent
React.lazy接受一个返回Promise的函数,这个Promise需要resolve一个export default React组件的模块
const OtherComponent = React.lazy(() => import('./OtherComponent'));
function MyComponent() {
return (
);
}
- 第二个需求,loading。使用
React.Suspense
实现。
Suspense
能够在包含其中的懒加载组件加载出来之前渲染给定的fallback组件
注意Suspense
可以位于懒加载组件之上的任意位置,Suspense
组件其中可以包含多个懒加载组件
const OtherComponent = React.lazy(() => import('./OtherComponent'));
function MyComponent() {
return (
Loading... }>