在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
随机推荐
- jQuery选择器之全选择器(*选择器)
在css中,经常会在第一行写下这样一段样式: * { margin:; padding:; } 通配符*意味着给所有的元素设置默认的边距.jQuery中我们也可以通过传递*选择器来选中文档页面中的元素 ...
- ccpc 网络赛 hdu 6155
# ccpc 网络赛 hdu 6155(矩阵乘法 + 线段树) 题意: 给出 01 串,要么询问某个区间内不同的 01 子序列数量,要么把区间翻转. 叉姐的题解: 先考虑怎么算 \(s_1, s_2, ...
- Android应用开发EditText文本内容变化监听方法
import android.app.Activity; import android.os.Bundle; import android.text.Editable; import android. ...
- 求LCA最近公共祖先的在线ST算法_C++
ST算法是求最近公共祖先的一种 在线 算法,基于RMQ算法,本代码用双链树存树 预处理的时间复杂度是 O(nlog2n) 查询时间是 O(1) 的 另附上离线算法 Tarjan 的链接: http ...
- nessus plugins 离线更新
1.打开 https://plugins.nessus.org/v2/offline.php 2.申请Activation Code http://www.tenable.com/products/n ...
- DropZone图片上传控件的使用
前台代码: <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w ...
- HTML5初学笔记
今天学习了下HTML5的基本知识,用画笔在画布上画了几个东西,效果如图,相关代码如下,注意点总结在末尾: <!DOCTYPE html> <html> <head> ...
- java如何增加数组长度
遇到一个面试题:在不使用list的add方法的情况下,动态的添加元素(大概是这个样子): ArrayList是基于数组实现的,是一个动态数组,其容量能自动增长,需学习arraylist的相关知识(ht ...
- Jquery操作基本筛选过滤器
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- springBoot Ribbon Hystrix Dashboard
1.引入依赖 <!-- 引入关于 hystrix Dashboard的依赖 --> <dependency> <groupId>org.springframewor ...