最近在做项目的时候遇到一个问题,在 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. [译]在python中如何有效的比较两个无序的列表是否包含完全同样的元素(不是set)?

    原文来源: https://stackoverflow.com/questions/7828867/how-to-efficiently-compare-two-unordered-lists-not ...

  2. JSP/Servlet Web 学习笔记 DayFour

    Servlet概述 Servelt是使用Java Servlet应用程序接口及相关类和方法的Java程序. Servlet是用Java编写的Server端程序,它与协议和平台无关.Servlet运行于 ...

  3. Android记事本开发03

    昨天: 生成签名文件及导出apk 遇到的问题: 无. 今天: activity和intent基础

  4. 团队冲刺Alpha(九)

    目录 组员情况 组员1(组长):胡绪佩 组员2:胡青元 组员3:庄卉 组员4:家灿 组员5:凯琳 组员6:翟丹丹 组员7:何家伟 组员8:政演 组员9:黄鸿杰 组员10:刘一好 组员11:何宇恒 展示 ...

  5. sql的over函数的使用

    over不能单独使用,要和分析函数:rank(),dense_rank(),row_number()等一起使用.其参数:over(partition by columnname1 order by c ...

  6. 不允许有匹配 "[xX][mM][lL]" 的处理指令目标。

    xml文件报错: 不允许有匹配 "[xX][mM][lL]" 的处理指令目标. 指的注意的是规范的XML格式:  <?xml version="1.0" ...

  7. transform perspective的层级问题

    如上图,在积分的数字元素上,使用了transform perspective,其层级就穿透了上面的遮罩层,关键代码如下: .mask { position: fixed; z-index:; } .f ...

  8. 洛谷 P2715 约数和

    给出a和b求a^b的约数和. 题目描述 输入输出格式 输入格式: 一行两个数a,b. 输出格式: 一个数表示结果对 9901 的模. 输入输出样例 输入样例#1: 2 3 输出样例#1: 15 说明 ...

  9. PriorityQueue详解(一)

    在Java SE 5.0中,引入了一些新的Collection API,PriorityQueue就是其中的一个.今天由于机缘巧合,花了一个小时看了一下这个类的内部实现,代码很有点意思,所以写下来跟大 ...

  10. 我们曾经心碎的C#之 第一章.我的第一个C#程序

    第一章.      C#入门 1.1        .NET与C#            001..NET是Microsoft.NET的简称,是基于Windows平台的一种技术            ...