在React组件unmounted之后setState的报错处理
最近在做项目的时候遇到一个问题,在 react 组件 unmounted 之后 setState 会报错。我们先来看个例子, 重现一下问题:
class Welcome extends Component {
state = {
name: ''
}
componentWillMount() {
setTimeout(() => {
this.setState({
name: 'Victor Wang'
})
}, 1000)
}
render() {
return <span>Welcome! {this.state.name}</span>
}
}
class WelcomeWrapper extends Component {
state = {
isShowed: true
}
componentWillMount() {
setTimeout(()=> {
this.setState({
isShowed: false
})
}, 300)
}
render() {
const message = this.state.isShowed ? <Welcome /> : 'Bye!'
return (
<div>
<span>{ message }</span>
</div>
)
}
}
举的例子不是很好,主要是为了说明问题。在 WelcomeWrapper 组件中, 300ms 之后移除了 Welcome 组件,但在 Welcome 组件里 1000ms 之后会改变 Welcome 组件的状态。这时候 React 会报出如下错误:
Warning: setState(...): Can only update a mounted or mounting component. This usually means you called setState() on an unmounted component.
这种错误情况一般出现在 react 组件已经从 DOM 中移除。我们在 react 组件中发送一些异步请求的时候, 就有可能会出现这样的问题。举个例子,我们在 componentWillMount 中发送异步请求,当请求成功返回数据,我们调用 setState 改变组件的状态。但是当请求到达之前, 我们更换了页面或者移除了组件,就会报这个错误。这是因为虽然组件已经被移除,但是请求还在执行, 所以会报setState() on an unmounted component
的错误。
决解问题
好了, 我们现在知道问题出现的原因, 我们该怎么解决这个问题?思路也很简单, 我们只要在 react 组件被移除之前终止 setState 操作就行了。回到之前的例子, 我们可以这样做:
componentWillMount() {
// 我们把 setTimeout 保存在 timer 里
this.timer = setTimeout(() => {
this.setState({
name: 'Victor Wang'
})
}, 1000)
}
// 在组件将要被移除的时候,清除 timer
componentWillUnmount() {
clearTimeout(this.timer)
}
类似的在处理 ajax 请求的时候也是这个套路, 在 componentWillUnmount 方法中终止 ajax 请求即可,以 jquery 为例:
componentWillMount() {
this.xhr = $.ajax({
// 请求的细节
})
}
componentWillUnmount() {
this.xhr.abort()
}
在处理 fetch 请求的时候思路也是一样的, 但是处理起来就没有那么容易了。 因为 Promise 不能被取消, 至少从目前的规范来看是没有相应的 API 来取消 Promise chain 的。将来可能会实现相应的 API, 感兴趣的可以看看这里和这里的讨论。
fecth 请求的处理
为了让 Promise 可以被取消,我们处理的思路是这样的,我们在我们的 Promise 外面再包裹一层 Promise 来保证我们的 Promise 可以被取消。下面看代码:
const makeCancelable = (promise) => {
let hasCanceled_ = false;
const wrappedPromise = new Promise((resolve, reject) => {
promise.then((val) =>
hasCanceled_ ? reject({isCanceled: true}) : resolve(val)
);
promise.catch((error) =>
hasCanceled_ ? reject({isCanceled: true}) : reject(error)
);
});
return {
promise: wrappedPromise,
cancel() {
hasCanceled_ = true;
},
};
};
这个 pattern 是由@istarkov提出来的。
上面用到 Promise 的相关知识, 不熟悉 Promise 的同学可以参考这里。
现在我们就可以用 makeCancelable 来取消我们的 fetch 请求了。
componentWillMount() {
// 为了简单和方便, 这里我用 setTimeout 来模仿一个需要很长时间的 fetch 请求
const mimicFetch = (resolve, reject) => {
setTimeout(() => {
resolve('Victor Wang')
}, 1000)
}
const promise = new Promise(mimicFetch)
this.cancelable = makeCancelable(promise)
this.cancelable.promise.then(name => {
this.setState({
name
})
}, (e) => {
console.log(e)
})
}
componentWillUnmount() {
// 在这取消
this.cancelable.cancel()
}
预防错误
为了避免这种错误的发生,我们有一个通用的 pattern 来处理这个问题。
componentWillMount () {
// add event listeners (Flux Store, WebSocket, document, etc.)
}
componentWillUnmount () {
// remove event listeners (Flux Store, WebSocket, document, etc.)
}
注:有什么不对的地方, 欢迎指正!
在React组件unmounted之后setState的报错处理的更多相关文章
- React使用antd按需引入报错
引言 按照antd官网配置按需引入,还是出现一系列的报错: 原因 在网上搜了一下,大部分说是react-scripts以及react-app-rewired版本不兼容的问题,我果断把下载低版本 npm ...
- react native get started run 模拟机报错解决
参照 http://reactnative.cn/docs/0.30/getting-started.html#content 1)当执行 react-native run-android 这个环节的 ...
- React Native真机红屏报错总结
一.run-android报错:unable to load script from assets 'index.android.bundl' 解决: 1.进入\android\app\src\mai ...
- 关于本地使用antd的upload组件上传文件,ngnix报错405的问题
使用阿里的ui框架antd的upload,会自动请求ngnix上面的一个路径,也就是action所在的位置,一直报错405 not allowed,后来经讨论,统一将action写成一个路径,后端对这 ...
- React中input框设置value报错解析
react input 不设置onChange的常见错误截图 表单是前端非常重要的一块内容,并且往往包含了错误校验等逻辑. React对表单元素做了专门的优化处理,他对表单元素做了一些抽象,使得他们 ...
- nodejs+react使用webpack打包时控制台报错
一.错误:Uncaught ReferenceError: process is not defined 解决方法: new webpack.DefinePlugin({ 'process.env': ...
- Vue 组件封装发布到npm 报错 Uncaught TypeError: Cannot read property 'toLowerCase' of undefined
Uncaught TypeError: Cannot read property 'toLowerCase' of undefined 原因是 没有导出 export default { name:& ...
- 多组件共享-vuex —— 使用vuex 报错 actions should be function or object with ”handler“
vuex分模块使用时出现的问题,单文件暂时没有用到 原因是在action 文件中没有任何定义(即:文件为空)或则 action 没有任何方法返回,将action在模块引用时去掉即可 转自:https: ...
- react 脚手架装后 运行eject报错 的 正确运行方式
git init git add . git commit -m 'init' npm run eject 或者 cnpm run eject
随机推荐
- 初识 HTML5(一)
H5其实就是H4的一个增强版本,我们在利用H5进行网页的构造会更简便,标签语义更简洁明了.首先,我们要理解HTML4,它是HTML的标记+css2+JavaScript的一些基本应用,简言之,就是AP ...
- 如何修改root密码
默认情况下,每次登录ubuntu都会生成一个随机的root密码,如果想要修改, sudo passwd 然后输入密码,这个密码就作为root用户的密码
- aFlex脚本入门
aFlex脚本入门 来源:http://blog.51cto.com/virtualadc/599194 来源:http://blog.51cto.com/virtualadc/624219 对于A1 ...
- 洛谷 P3747 [六省联考2017]相逢是问候 解题报告
P3747 [六省联考2017]相逢是问候 题目描述 \(\text {Informatik verbindet dich und mich.}\) 信息将你我连结. \(B\) 君希望以维护一个长度 ...
- python的pip安装
http://blog.csdn.net/liuchunming033/article/details/39578019
- html模板引擎jade的使用
jade语法: #{xxx} //嵌入数据 p= xxx //嵌入数据 p #{xx} //嵌入数据 标签 html // 翻译为<html></html> div#test ...
- hdu 4359 dp
/* 题目大意:给n个节点的二叉树第i个节点的权值为2^(i-1), 求所有含左右子树的节点都符合左子树的权和小于右子树权和的种数. */ #include <iostream> #inc ...
- [9018_1563][bzoj_2144]跳跳棋
题目描述 Hzwer的跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子. 某一天,黄金大神和cjy用跳跳棋来做一个简单的游戏:棋盘上有3颗棋子,分别在a,b,c这三个位置.他们 ...
- bzoj 3744 Gty的妹子序列 区间逆序对数(在线) 分块
题目链接 题意 给定\(n\)个数,\(q\)个询问,每次询问\([l,r]\)区间内的逆序对数. 强制在线. 思路 参考:http://www.cnblogs.com/candy99/p/65795 ...
- VIM使用系列: 复制并移动文本
1 5. 复制并移动文本 *copy-move* 2 3 *quote* 4 "{a-zA-Z0-9.%#:-"} 指定下次的删除.抽出和放置命令使用的寄存器 5 {a-zA-Z0 ...