React 函数组件中对window添加事件监听resize导致回调不能获得Hooks最新状态的问题解决思路
React 函数组件中对window添加事件监听resize导致回调不能获得Hooks最新状态的问题解决思路
这几天在忙着把自己做的项目中的类组件转化为功能相同的函数组件,首先先贴一份该组件类组件的关键代码:
import React, {Component} from "react"
// 防抖组件
import { debounce } from 'throttle-debounce';
// 引入connect用于连接UI组件与redux
import {connect} from "react-redux"
class HandSendMsg extends Component {
state = {
w_width:window.innerWidth,
w_height:window.innerHeight,
}
// 向UE4像素流发送消息
emitMessageToUE4 = (msgObj) => {
// this.props.video_ref证明已经挂载了UE4像素流推送标签组件
if(this.props.video_ref && this.props.video_ref.current && this.props.video_ref.current.emitMessage instanceof Function) {
this.props.video_ref.current.emitMessage(msgObj);
}
}
// 修改分辨率的函数回调
handleResize = () => {
let w_width = window.innerWidth
let w_height = window.innerHeight
this.setState({w_width, w_height})
this.emitMessageToUE4({
time:Date.now(),
width:w_width,
height:w_height,
funcName:"onResize"
})
}
componentDidMount() {
// 给全局绑定一个修改分辨率的操作, 当窗口发生变化的时候就修改分辨率,加入了防抖操作
window.addEventListener("resize", debounce(300, false, this.handleResize.bind(this)))
}
render() {
const {w_height, w_width, iconLocList } = this.state
return (
<div>
</div>
)
}
}
// 使用connect()()创建并暴露一个Count的容器组件
export default connect(
state => ({
signal_state:state.signal,
video_ref:state.videoRef
}),
{
}
)(HandSendMsg)
然后我试着改写成函数组件,初始关键代码是这样的:
import React, { useState, useEffect } from 'react'
// 防抖组件
import { debounce } from 'throttle-debounce';
// 引入useSelector用于读取redux store中保存的数据
import { useSelector } from "react-redux"
export default function HandSendMsg() {
// state数据
const [wh, setWh] = useState({
width:window.innerWidth,
height:window.innerHeight
})
// 标志是否开始监听UE4消息,确保只监听一次
const [start_listen, setStart_listen] = useState(false)
// store里保存的数据
const signal_state = useSelector(state => state.signal)
const video_ref = useSelector(state => state.videoRef)
// 向UE4像素流发送消息
const emitMessageToUE4 = (msgObj) => {
// video_ref不为null证明已经挂载了UE4像素流推送组件
if(video_ref && video_ref.current && video_ref.current.emitMessage instanceof Function) {
video_ref.current.emitMessage(msgObj);
}
}
// 修改分辨率的函数回调
function handleResize() {
let w_width = window.innerWidth
let w_height = window.innerHeight
emitMessageToUE4({
time:Date.now(),
width:w_width,
height:w_height,
funcName:"onResize"
})
setWh({
width:w_width,
height:w_height
})
}
useEffect(() => {
// 给全局绑定一个修改分辨率的操作, 当窗口发生变化的时候就修改分辨率
window.addEventListener('resize',debounce(300,false,handleResize))
// eslint-disable-next-line
}, [])
return (
<div>
</div>
)
}
useState钩子替代类组件中的state,useEffect副作用钩子替代类组件中的生命周期钩子(componentDidMount(), componentDidUpdate()和 componentWillUnmount()), useSelector和useDispatch来取代redux中的connect,这些都没什么好说的,但是在我给window添加onresize的事件监听的时候, 却发现只能获取到hooks的初始值,里面函数的调用根本无法获得最新的hooks值,无法更新state值:
window.addEventListener('resize',debounce(300,false,handleResize))
于是我在return里加了几个button,一个button用来获取最新的hooks的值,一个button用来调用window的onresize所调用的handleResize函数,发现在外面可以进行最新hooks的获取和更新,如下图所示:
在费劲周折之后,在网上看到帖子说可以对这个useEffect进行优化,使其监听一个变量,从而来使事件的绑定有效,我想了一下,要想使信息发送过去,我必须得确保signal_state这个值为true,那么我就用它来进行监听吧,当其从false改为true的时候再开启onResize的事件监听,问题不就解决了么。然后实践了一下,果然可以了,关键代码如下:
import React, { useState, useEffect } from 'react'
// 防抖组件
import { debounce } from 'throttle-debounce';
// 引入useSelector用于读取redux store中保存的数据
import { useSelector } from "react-redux"
export default function HandSendMsg() {
// state数据
const [wh, setWh] = useState({
width:window.innerWidth,
height:window.innerHeight
})
// 标志是否开始监听UE4消息,确保只监听一次
const [start_listen, setStart_listen] = useState(false)
// store里保存的数据
const signal_state = useSelector(state => state.signal)
const video_ref = useSelector(state => state.videoRef)
// 向UE4像素流发送消息
const emitMessageToUE4 = (msgObj) => {
// video_ref不为null证明已经挂载了UE4像素流推送组件
if(video_ref && video_ref.current && video_ref.current.emitMessage instanceof Function) {
video_ref.current.emitMessage(msgObj);
}
}
// 修改分辨率的函数回调
function handleResize() {
let w_width = window.innerWidth
let w_height = window.innerHeight
emitMessageToUE4({
time:Date.now(),
width:w_width,
height:w_height,
funcName:"onResize"
})
setWh({
width:w_width,
height:w_height
})
}
useEffect(() => {
// 给全局绑定一个修改分辨率的操作, 当窗口发生变化的时候就修改分辨率
// 只有signal_state即信令状态由false改为true的时候才添加一次事件
if(signal_state) {
window.addEventListener('resize',debounce(300,false,handleResize))
}
// eslint-disable-next-line
}, [signal_state])
return (
<div>
</div>
)
}
至此,问题已经解决了,总体来说函数式组件由于市面上教程较少,坑也比较多,这次这个给window添加监听事件的需要有一个变量同时监听变化,使得useEffect()可以正确响应然后重新绑定监听,为什么要使用useEffect()呢?因为我们的需求是必须确保只给window添加一次事件监听,当然得要求是对的了,而useEffect()可以模拟componentDidMount(), 可以最快的解决。
如果大家还有什么疑问或者新的建议,希望大家可以在下面留言,我看到一定会回复的,谢谢!!!
React 函数组件中对window添加事件监听resize导致回调不能获得Hooks最新状态的问题解决思路的更多相关文章
- vue中给window添加滚动监听无效的解决方案
原文链接: 点我 页面中有这么一个需求,当页面滚动到一定高度之后,页面中的某些元素进行吸顶,固定到顶部位置,或者是滚动到一定程度进行更新数据的操作.我相信不少网友查阅过类似的资料,网友给出的解决方案, ...
- js添加事件监听的方式与this
js添加事件监听与this js添加事件监听的方式与this 在标签中调用自定义函数 DOM0级事件处理程序 DOM2级事件处理程序 this 代表谁? js添加事件监听的方式与this <di ...
- GridView添加事件监听和常用属性解析
1. 使用流程 graph LR 准备数据源-->新建适配器 新建适配器-->绑定数据源 绑定数据源-->加载适配器 2. 常用属性 android:columnWidth:每一列的 ...
- extjs组件添加事件监听的三种方式
extjs对组件添加监听的三种方式 在定义组件的配置时设置 如代码中所示: Java代码 xtype : 'textarea', name : 'dataSetField', labelSe ...
- vuejs2.0实现分页组件,使用$emit进行事件监听数据传递
上一篇文章介绍了vuejs实现的简单分页,如果我有几个页面都需要有分页效果,不可能每个页面都去复制一下这段代码吧,意思是封装一下,变成通用的组件. 首先使用基础 Vue 构造器,创建一个“子类”,Vu ...
- [转]extjs组件添加事件监听的三种方式
原文地址:http://blog.csdn.net/y6300023290/article/details/18989635 1.在定义组件配置的时候设置 xtype : 'textarea', na ...
- Android实战简易教程-第十五枪(实现ListView中Button点击事件监听)
1.main.xml <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" x ...
- 添加事件监听兼容IE6-8
IE8一下浏览器不支持addEventListener,用attachEvent取而代之,但是在时间类型前面要加上’on‘,例如click时间在attachEvent中要写成onclick. var ...
- 使用addeventlistener为js动态创建的元素添加事件监听
点击li弹出内容,并且动态添加li之后有效 <button onclick="addFunction()">点我增加</button> <ul> ...
随机推荐
- Solon Web 开发,十二、统一的渲染控制
Solon Web 开发 一.开始 二.开发知识准备 三.打包与运行 四.请求上下文 五.数据访问.事务与缓存应用 六.过滤器.处理.拦截器 七.视图模板与Mvc注解 八.校验.及定制与扩展 九.跨域 ...
- 分布式一致性协议Raft,以及难搞的Paxos
https://blog.csdn.net/colorant/article/details/73887706
- @WebServlet注解(Servlet注解)
@WebServlet 注解的属性 @WebServlet 用于将一个类声明为 Servlet,该注解会在部署时被容器处理,容器根据其具体的属性配置将相应的类部署为 Servlet.该注解具有下表给出 ...
- 返回值Object-注解驱动作用
返回值Object 两个常用实现类 StringHttpMessageConverterhe:负责读取字符串格式的数据和写出字符串格式的数据 MappingJackson2HttpMessageCon ...
- Clang-Format 个人常用配置
Clang-Format 个人常用配置 本文记录 Clang-Format 个人常用配置. 欲了解更多配置选项,可查阅 官方文档. BasedOnStyle: Google AccessModifie ...
- Device or resource busy
格式化磁盘显示忙碌,如何解决呢? [root@jp33e503-11-8 ~]# mkfs.xfs /dev/sdc mkfs.xfs: cannot open /dev/sdc: Device or ...
- 图论+回溯解QQ一笔画红包
[春节整活] QQ的一笔画红包有几个特性: 1.最大为5×5的点阵,所以可以把每个点从左到右,从上到下标为1-25号点 2.每两个点只能存在一条线 3.线可以被盖住(例如连接2-1-3,2-1的线会被 ...
- Java 后台线程介绍
一 是啥? package com.aaa.threaddemo; /* * 一 Java后台线程? * 守护线程--也称"服务线程",他是后台线程, * 它有一个特性,即为用户 ...
- Three.js 实现2022冬奥主题3D趣味页面 🐼
背景 迎冬奥,一起向未来!2022冬奥会马上就要开始了,本文使用 Three.js + React 技术栈,实现冬日和奥运元素,制作了一个充满趣味和纪念意义的冬奥主题 3D 页面.本文涉及到的知识点主 ...
- Servlet Servlet的装载三种情况
感谢原文作者:DaleyDC 原文链接:https://blog.csdn.net/sinat_32873711/article/details/53170342 Servlet的装载三种情况: 自动 ...