最近在做项目的时候遇到一个问题,在 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的报错处理的更多相关文章

  1. React使用antd按需引入报错

    引言 按照antd官网配置按需引入,还是出现一系列的报错: 原因 在网上搜了一下,大部分说是react-scripts以及react-app-rewired版本不兼容的问题,我果断把下载低版本 npm ...

  2. react native get started run 模拟机报错解决

    参照 http://reactnative.cn/docs/0.30/getting-started.html#content 1)当执行 react-native run-android 这个环节的 ...

  3. React Native真机红屏报错总结

    一.run-android报错:unable to load script from assets 'index.android.bundl' 解决: 1.进入\android\app\src\mai ...

  4. 关于本地使用antd的upload组件上传文件,ngnix报错405的问题

    使用阿里的ui框架antd的upload,会自动请求ngnix上面的一个路径,也就是action所在的位置,一直报错405 not allowed,后来经讨论,统一将action写成一个路径,后端对这 ...

  5. React中input框设置value报错解析

    react input 不设置onChange的常见错误截图 表单是前端非常重要的一块内容,并且往往包含了错误校验等逻辑.  React对表单元素做了专门的优化处理,他对表单元素做了一些抽象,使得他们 ...

  6. nodejs+react使用webpack打包时控制台报错

    一.错误:Uncaught ReferenceError: process is not defined 解决方法: new webpack.DefinePlugin({ 'process.env': ...

  7. Vue 组件封装发布到npm 报错 Uncaught TypeError: Cannot read property 'toLowerCase' of undefined

    Uncaught TypeError: Cannot read property 'toLowerCase' of undefined 原因是 没有导出 export default { name:& ...

  8. 多组件共享-vuex —— 使用vuex 报错 actions should be function or object with ”handler“

    vuex分模块使用时出现的问题,单文件暂时没有用到 原因是在action 文件中没有任何定义(即:文件为空)或则 action 没有任何方法返回,将action在模块引用时去掉即可 转自:https: ...

  9. react 脚手架装后 运行eject报错 的 正确运行方式

    git init git add . git commit -m 'init' npm run eject 或者 cnpm run eject

随机推荐

  1. HDU 3856 Palindrome ( Manacher + RMQ + 二分 ) WA!!!

    不知道错在哪了,求大神指教!!! 思路:用manacher求出每个以str[i]为中心轴的回文串的长度,RMQ预处理区间最大值,对于每个查询,二分最大回文串长,判定是否可行. #include < ...

  2. ubuntu wifi连接出现Network service discovery disabled的解决办法

    修改/etc/default/avahi-daemon,将AVAHI_DAEMON_DETECT_LOCAL从1改为0(关闭avahi) ------------------------------- ...

  3. 国际语言代码 Language Code

    语言代码 语言名称 af 南非语 af-ZA 南非语 ar 阿拉伯语 ar-AE 阿拉伯语(阿联酋) ar-BH 阿拉伯语(巴林) ar-DZ 阿拉伯语(阿尔及利亚) ar-EG 阿拉伯语(埃及) a ...

  4. java GUI Graphics2D 绘图

    Graphics类提供基本绘图方法,Graphics2D类提供更强大的绘图能力.本节讲解Graphics类,下节讲解Graphics2D. Graphics类提供基本的几何图形绘制方法,主要有:画线段 ...

  5. UltraEdit 删除空行

    UltraEdit 删除空行 数据里有大量的空行,想在UltraEdit里删除,在网上搜了很多方法都不管用,功夫不负有心人,最后终于找到了可用的方法: 搜索—>替换,在“查找什么”里输入:\n( ...

  6. org.apache.catalina.core.StandardWrapperValve.invoke Servlet.service() for servlet [WebApp] in context with path关于数据库库的问题

    1.在本地~~把你的查询的sql打印出来~~~然后拿到测试库里面执行一遍~~然后拿到正式库里面在执行一遍  2.看生产和本地~~1).代码不同步:2).就是数据为空问题 3.也有可能是数据库配置文件问 ...

  7. em,rem

    em rem 相对单位:  也可用于设置padding line-height等em相对当前容器的默认字体设置比如,所有浏览器默认字体都是16px,body{ font-size:62.5%}以后即1 ...

  8. 飞思卡尔MC9S12系列单片机地址影射以及分页问题

    对于用MCU的人来说,不一定要明白HCS12(x) memory map的机制和联系.因为如果没有系统地学习操作系统和编译原理之类的课程,确实有些难度.并且,对于DG128 XS128这样的MCU,默 ...

  9. 各版本Sql Server下载地址全

    SQL Server 2014简体中文企业版 文件名:cn_sql_server_2014_enterprise_edition 32位下载地址:ed2k://|file|cn_sql_server_ ...

  10. [LeetCode] Sort List 排序 sort

    Sort a linked list in O(n log n) time using constant space complexity. Hide Tags Linked List Sort   ...