react 函数组件的缺点

1. 容易产生闭包

若在useEffect中使用监听事件addEventListener调用func方法,js会记录此时的上下文数据,当调用func方法时,使用的数据是旧的数据

解决方法:可以在data数据获取到之后再添加监听,或者data数据变化就重新添加监听,但是这样会可能导致无法取消监听的情况,越加越多,每次都会调用几次。

2. 无法取消监听

若在useEffect中使用了监听addEventListener调用func方法,由于每次渲染都会重新调用运行函数组件,所以func每次的函数引用都会更新,会导致无法取消监听removeEventListener

解决方法:使用useCallback,第二个参数为空“[]”(不随任何变化而变化),但是这样又回到第一个问题,函数会形成闭包,里面使用的数据还是旧数据

function App(props){
    const [data, setData] = useState({})
    const func = () => {
        console.log(data) // 即使data变化了,输出的还是 “{}”
        ajax("url", {id: data.id}) // data.id为undefined
    }
    useEffect(() => {
      if(props.visible){
        window.addEventListener("message", func)
      }else {
        window.removeEventListener("message", func) // 若组件重新渲染过,func会重新定义,函数的引用变化了,所以监听会无法取消
      }
    }, [props.visible])
}

// 使用类组件可以避免这2种情况
class App extends Component {
  componentDidUpdate(prevProps){
    if(!prevProps.visible && this.props.visible){ // 打开
      window.addEventListener("message", this.func)
    }else if(prevProps.visible && !this.props.visible){ // 关闭
      window.removeEventListener("message", this.func)
    }
  }
  // 或者使用componentDidMount、componentWillUnMount
  componentDidMount(){
      window.addEventListener("message", this.func)
  }
  componentWillUnMount(){
      window.removeEventListener("message", this.func)
  }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值