翻了波之前写的文章还有笔记,发现关于前端的文章并不多(好歹也划水做过点前端开发)。巧了,最近没什么好话题可写,做下 React Hooks 学习笔记吧。

Effect Hook

不得不说 Hook 的出现降低了我们在 React 中处理副作用(side effect)的心智负担,通过 useEffect 就可以很好的完成之前需要使用几个生命周期函数配合才能完成的事。

Effect Hook 死循环请求问题

由于 Effect Hook 不熟「官方文档没读透」,最近使用 useEffect 出现了异步请求发送了无限次的问题,翻🚗了。我有个组件大概是这么写的:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
import React, { useState, useEffect } from 'react';

import request from 'umi-request';

import logo from './logo.svg';
import './App.css';

function App() {
const [data, setData] = useState([]);

useEffect(() => {
request('https://jsonplaceholder.typicode.com/todos/', {
method: 'get',
})
.then(response => {
console.log('fetch data');
setData(response);
})
.catch(error => {
console.log("report error: ", error);
})
});

return (
<div className="App">
<header className="App-header">
<img src={logo} className="App-logo" alt="logo" />
<ul>
{data.map(item => {
return (
<li key={item.id}>{item.title}</li>
);
})}
</ul>
</header>
</div>
);
}

export default App;

效果如下:

效果

https://zo2c2.csb.app/,可以很方便的从调试控制台看到,异步请求一直在发,陷入了死循环之中。这是为什么?因为 useEffect 会在组件 Mounting 和 Updating 阶段执行。每次 request 请求成功,我们都会设置一次组件的 state -> data,所以组件会更新,useEffect 会再次执行,循环往复,造成了无限重复请求问题。那么,如何解决这个问题?之前我忽略了 useEffect 第二个参数的存在,使用 useEffect 的第二个参数可以解决这个问题。一般情况下,我们希望组件只在 mounting 阶段异步获取数据,所以,我们可以这么设置 useEffect 的第二个参数,让它具有和 componentDidMount 生命周期函数类似的行为(组件第一次 mount 后执行):

React 组件生命周期

1
2
3
4
5
6
7
8
9
10
11
12
useEffect(() => {
request('https://jsonplaceholder.typicode.com/todos/', {
method: 'get',
})
.then(response => {
console.log('fetch data');
setData(response);
})
.catch(error => {
console.log("report error: ", error);
})
}, []);

我们传递了一个空数组作为 useEffect 的第二个参数,这样就能避免在组件 Updating 阶段执行 useEffect。这个数组成为依赖数组。依赖数组为空,表明 useEffect 不会因为某个变量的变化而再次执行。在组件需要根据某个变量变化进行渲染的时候,可以将此变量放到依赖数组中,一旦这个依赖的变量变动,useEffect 就会重新执行。

让组件卸载后做点事

在 class 组件中,我们可以将组件卸载后要做的事放在 componentWillUnmount 中。引入 Hook 后,在 function 组件中,我们可以把组件卸载要做的事放在 useEffect 中,让它返回一个 callback 即可,如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
import React, { useState, useEffect } from "react";

function Child({ visible }) {
useEffect(() => {
alert("组件已挂载");
return () => { // return 一个 callback
alert("组件已被卸载!!");
};
}, []);

return visible ? "true" : "false";
}

export default function App() {
const [visible, changeVisible] = useState(true);

return (
<div>
{visible && <Child visible={visible} />}
<button
onClick={() => {
changeVisible(!visible);
}}
>
{visible ? "卸载组件" : "挂载组件"}
</button>
</div>
);
}

effect 卸载组件做点事儿

参考

评论