Vue转React开发经验分享——hooks写法如何触发react生命周期、如何触发数据更新?
背景:习惯了vue的写法,并且vue2和vue3都比较熟悉,在转react开发中,不停的思考react和vue框架的相似之处,以及vue中的写法在react里怎么替换。本文将组件更新或组件生命周期角度出发聊聊如何使用hooks触发生命周期,以及如何触发数据更新
为什么要触发组件生命周期
很多没有经验的人不知道为什么要学习框架的生命周期。在开发中,我们经常需要弹窗和主页面进行数据传递,我现在就遇到很多组件通信的场景,主页面更新了,弹窗不能关闭,弹窗也要更新;又或者,弹窗更新了,立即更新外部页面,如tree的状态,视图更新等。这里先不考虑组件通信,之后在聊。先说组件如何触发更新。
组件什么时候更新?这里我们说的概念是主动触发更新。当你明确知道哪个数据变化了,一定要更新;或者执行了变更操作一定要更新我们就去触发更新。react的hook写法没有提供像vue这种onMounted,onUnMounted,watch等监控方法,同时它的数据必须通过组件更新才能重新渲染到视图。所以在用react开发的时候,更多的是在跟数据打交道,有时候莫名奇妙的数据变成undefined,或者数据更新了视图不更新。
useEffect
useEffect
是 React 中最常用的生命周期管理 Hook。它相当于类组件中的componentDidMount
、componentDidUpdate
和componentWillUnmount
的组合。
useEffect模拟组件挂载和卸载
最常用的就是useEffect方法,这个方法能干什么?首先,它接受两个参数,第一个参数放执行逻辑,第二个参数放依赖数组。同时第一个参数的函数返回支持一个return方法。这几项共同演绎了组件更新逻辑。首先,如何模拟组件挂载和卸载,比如弹窗如果封装为一个组件,那么弹窗打开是挂载,弹窗关闭是卸载。你如果只想让组件执行一次挂载和卸载逻辑,就将依赖项设置一个空数组。在第一个参数写组件挂载逻辑,在返回return里执行卸载,即清理工作。
示例如下
import React, { useEffect } from 'react';function Example() {useEffect(() => {// 这里执行的代码在组件首次渲染后只执行一次const interval = setInterval(() => {console.log('Interval running');}, 1000);// 返回一个清理函数,用于在组件卸载时清除定时器return () => {clearInterval(interval);console.log('Component unmounted');};}, []);return (<div>{/* 组件内容 */}</div>);
}
Vue中的
onMounted
,可以在首次渲染后执行一些初始化操作。在React中,你可以通过useEffect
的依赖项为空来实现这一点;Vue中的
onUnmounted
,可以在组件卸载前执行一些清理操作;推荐使用useEffect
的返回值来实现更安全的组件卸载逻辑
useEffect模拟数据更新视图
在Vue中,组件的更新操作通常是由数据变化触发的,Vue会自动处理DOM的更新。而在React中,如果你想模拟Vue的组件更新操作,你可以使用
useState
和useEffect
钩子来手动控制组件的更新。通过useEffect设置依赖项,就可以在依赖项变化时触发其他数据的方法。比如删除数据后触发list刷新,保存数据后重新获取数据等。
import React, { useState, useEffect } from 'react';function Example() {const [count, setCount] = useState(0);useEffect(() => {// 组件更新时执行console.log(`Count changed to ${count}`);};}, [count]); // 依赖数组,当 count 变化时,重新执行 useEffectreturn (<div><p>Count is: {count}</p><button onClick={() => setCount(count + 1)}>Increment</button></div>);
}
如何触发数据更新
在vue中可以通过ref、reactive等方法直接定义响应式数据,数据变化视图自动更新。但是在react,数据需要手动管理,即哪些是需要视图跟着一起变化的数据需要手动维护。最常用的是useState钩子函数。这个函数通过解构重命名为[变量,变量的修改方法]。用法如下
useState
import React, { useState } from 'react';function MyComponent() {const [count, setCount] = useState(0);const handleClick = () => {setCount(count + 1);};return (<div><p>Count: {count}</p><button onClick={handleClick}>Increment</button></div>);
}
setState后异步获取state不是最新的?
如果在一个组件生命周期内,先通过方法更改了state,但是紧跟着就使用state,会出现用的还是旧的state数据。这是因为state还没更新,setState是异步的,state会在下个组件生命周期更新。所以你会发现你想的数据总是”晚一步“。这点非常不好用!!!如何避免?
解决方法
1.更新state时,用一个额外的变量存当前最新的state,可以用useRef维护一个全局变量,直接使用全局变量就行了;也可以用一个局部变量,然后立即使用这个局部变量
useRef
useRef
是React提供的一个Hook,用于创建一个保持初始值的可变引用。你可以在组件的函数中使用它来存储当前的状态值。
import React, { useState, useRef } from 'react';function MyComponent() {// 使用useState创建状态const [count, setCount] = useState(0);// 使用useRef创建一个引用,用于存储当前的count值const currentCount = useRef(count);// 当count更新时,也更新当前的引用React.useEffect(() => {currentCount.current = count;}, [count]);return (<div><p>当前计数:{currentCount.current}</p><button onClick={() => setCount(count => count + 1)}>增加计数</button></div>);
}export default MyComponent;
在函数组件中,你可以创建一个局部变量并立即使用它来存储当前的状态值。这样,你可以立即使用这个局部变量,而不用担心状态的延迟更新问题。
import React, { useState } from 'react';function MyComponent() {const [count, setCount] = useState(0);// 创建一个局部变量来存储当前的count值
const update=()=>{
const new = count+1;setCount(new);
//需要立即使用最新countupdateOther(new);
}
const updateOther =(count:number)=>{
//巴拉巴拉
}return (<div><p>当前计数:{currentCount}</p><button onClick={update}>增加计数</button></div>);
}export default MyComponent;
setState更新对象不成功?
在React中,当使用setState
更新对象时,如果直接修改状态对象,可能会导致原始数据丢失。这是因为React的setState
函数在内部使用的是不可变数据结构,它会合并新的状态而不是替换整个对象。
以下是一个可能导致原始数据丢失的例子:
setState({age: prevState.userInfo.age + 1
});
在这个例子中,你首先获取了prevState.userInfo
,然后直接在同一个对象上修改了age
属性。这样做不会创建一个新的对象,因此React不会检测到状态的变化,所以界面不会更新。
解决方法
为了避免这种情况,你应该始终创建一个新的状态对象,如下所示:
setState({...prevState.userInfo,age: prevState.userInfo.age + 1
});
通常用展开运算符复用老数据,将需要修改的属性进行变更。
相关文章:
Vue转React开发经验分享——hooks写法如何触发react生命周期、如何触发数据更新?
背景:习惯了vue的写法,并且vue2和vue3都比较熟悉,在转react开发中,不停的思考react和vue框架的相似之处,以及vue中的写法在react里怎么替换。本文将组件更新或组件生命周期角度出发聊聊如何使用hooks触发生命周期&…...

算法入门-贪心1
第八部分:贪心 409.最长回文串(简单) 给定一个包含大写字母和小写字母的字符串 s ,返回通过这些字母构造成的最长的回文串 的长度。 在构造过程中,请注意 区分大小写 。比如 "Aa" 不能当做一个回文字符串…...

element-plus的面包屑组件el-breadcrumb
面包屑组件主要用来显示当页面路径,以及快速返回之前的页面。 涉及2个组件 el-breadcrumb 和el-breadcrumb-item, el-breadcrumb的spearator指定item的分隔符 el-breadcrumb-item的to和replace属性和vue-router的一致,需要结合vue_router一起使用 用法…...
推荐几个网盘资源站给大伙,找资源更方便
夸克网盘在当前已然成为极为主流的网盘之一,其功能体验堪称强大,不仅支持在线解压阅读,磁力离线等功能也十分出色。那么,究竟该如何寻找夸克资源呢?下面,我就来为大家分享几个堪称神级的夸克资源网站。 一、…...

【Qt】Qml界面中嵌入C++ Widget窗口
1. 目的 qml做出的界面漂亮,但是执行效率低,一直想找一个方法实现qml中嵌入c界面。现在从网上找到一个方法,简单试了一下貌似可行,分享一下。 2. 显示效果 3. 代码 3.1 工程结构 3.2 pro文件 需要添加widgets > QT quick …...
Python快速入门 —— 第五节:接口开发
第五节:接口开发 目标: 学习使用Flask框架开发简单的Web接口,实现对学生信息的增删改查,通过HTTP请求与应用交互。 内容: Flask简介: Flask是一个轻量级的Python Web框架,使用简单,扩展性强,适合快速开发Web应用。安装Flask: pip install flask创建Flask应用: fr…...

利用secureCRT向虚拟机发送文件(secureCRT安装使用教程)
链接: secureCRT 链接:https://pan.baidu.com/s/1CvNYzoBbLVkyYNFq7hrT0g 提取码:5974 链接: secureCRT安装使用教程 链接:https://pan.baidu.com/s/1Bbi7SqyJBere8G53BCYL5A 提取码:xjw1...
AI杂七杂八系列(1)——工程篇
1. 远程服务器无法登录问题 2. 内存溢出解决方法 3. Padding 4. try...except...处理异常报错 5. view、expand、repeat、transpose、permute和squeeze、unsqueeze的区别 1. 远程服务器无法登录问题 权限可能是root权限,修改权限 用户权限: sudo c…...

学习大数据DAY58 增量抽取数据表
作业 1 SQL 优化的常见写法有哪些 - 面试经常被问 使用索引:合理创建和使用索引是提高查询效率的关键。索引可以加速数据的检 索速度,但是索引也会占用额外的存储空间,并且在插入、删除和更新操作时会 有额外的开销。 避免全表扫描&…...

HTTPTomcat
HTTP&Tomcat&Servlet 今日目标: 了解JavaWeb开发的技术栈理解HTTP协议和HTTP请求与响应数据的格式掌握Tomcat的使用掌握在IDEA中使用Tomcat插件 1,Web概述 1.1 Web和JavaWeb的概念 Web是全球广域网,也称为万维网(www),…...

Python数据分析-Matplotlib快速入门
一、pyplot 二、绘图 1.绘制x和y的点 2.无线绘图 3.多点 4.默认x点 三、标记 1.标记 2.参考 3.格式化字符串 4.尺寸 5.颜色 四、线条 1.线形 两个都是设置虚线 2.更短的语法 3.线参考 4.线条颜色 5.线宽度 6.多条线 也可以 五、标签 1.为绘图创建标签 2.为绘图设置标题 3…...
重塑在线软件开发新纪元:集成高效安全特性,深度解析与评估支持浏览器在线编程的系统架构设计
目录 案例 【题目】 【问题 1】(13 分) 【问题 2】(12 分) 【答案】 【问题 1】解析 【问题 2】解析 相关推荐 案例 阅读以下关于软件架构设计与评估的叙述,回答问题1和问题2。 【题目】 某公司拟开发一套在线软件开发系统,支持用户通过浏览器…...
【鸿蒙OH-v5.0源码分析之 Linux Kernel 部分】003 - vmlinux.lds 链接脚本文件源码分析
【鸿蒙OH-v5.0源码分析之 Linux Kernel 部分】003 - vmlinux.lds 链接脚本文件源码分析 系列文章汇总:《鸿蒙OH-v5.0源码分析之 Uboot+Kernel 部分】000 - 文章链接汇总》 本文链接:《【鸿蒙OH-v5.0源码分析之 Linux Kernel 部分】003 - vmlinux.lds 链接脚本文件源码分析》 …...
MongoDB实现高级RAG:Parent-Document检索技术详解
MongoDB实现高级RAG:Parent-Document检索技术详解 引言 在人工智能和自然语言处理领域,检索增强生成(Retrieval-Augmented Generation, RAG)技术正在迅速发展。本文将介绍一种更高级的RAG实现方式:Parent-Document检索。我们将探讨如何使用…...
胡学乱想----前端知识点(css色彩)
1. margin 属性 简写 margin 属性有两个值时,它将 margin-top 和 margin-bottom 设置为第一个值,并将 margin-left 和 margin-right 设置为第二个值 .marker {width: 200px;height: 25px;background-color: red;margin: 10px auto; }2. rgb 属性 CSS 的 rgb 函数接收红色…...
GEE 案例——利用MODIS数据和NDWI指数进行美国五大湖水体计算和时序分析(直方图统计和面积统计)
目录 简介 MODIS数据 代码 结果 简介 利用MODIS数据和NDWI指数进行水体计算和时序分析(直方图统计和面积统计),这里我们统计了2001-2023年的美国五大湖的水域面积变化情况。 MODIS数据 MODIS/061/MOD09A1数据是由美国宇航局(NASA)的Moderate Resolution Imaging Spe…...

【jvm】记一次hive堆heap内存溢出的排查
先看下java的内存模型 监控jvm工具:visualVM 摘录一下内容: 由c开发的jvm,它巧妙地设计了java的设计理念——即万物皆对象。并设计了这些对象应该如何存储,如何调用,并通过不断迭代设计让对象的存储和回收࿰…...

编译运行 webAssembly(wasm)
环境准备: lunix下docker 参考https://hub.docker.com/r/emscripten/emsdk 拉编译环境 docker pull emscripten/emsdk 编译 随便找个目录,敲下面命令,编译一个webAssembly 程序 # create helloworld.cpp cat << EOF > hellowo…...
Linux bash 关联数组
目录 一. 关联数组定义二. 访问关联数组三. 元素的添加与删除四. 键值对的获取与遍历五. 实际应用5.1 读取封装配置文件内容5.2 收集系统信息 一. 关联数组定义 从 Bash 4.0 开始,Bash 支持关联数组。关联数组允许你将键和值配对,并通过键来访问值&…...

选择排序
一:基本思想 每一次从待排序的数据元素中选出最小(或最大)的一个元素,存放在序列的起始位置,直到全部待排序的数据元素排完 。 解释:就是不断的找到最小的放在最左面,然后缩短数组,…...
golang循环变量捕获问题
在 Go 语言中,当在循环中启动协程(goroutine)时,如果在协程闭包中直接引用循环变量,可能会遇到一个常见的陷阱 - 循环变量捕获问题。让我详细解释一下: 问题背景 看这个代码片段: fo…...
【ROS】Nav2源码之nav2_behavior_tree-行为树节点列表
1、行为树节点分类 在 Nav2(Navigation2)的行为树框架中,行为树节点插件按照功能分为 Action(动作节点)、Condition(条件节点)、Control(控制节点) 和 Decorator(装饰节点) 四类。 1.1 动作节点 Action 执行具体的机器人操作或任务,直接与硬件、传感器或外部系统…...
ffmpeg(四):滤镜命令
FFmpeg 的滤镜命令是用于音视频处理中的强大工具,可以完成剪裁、缩放、加水印、调色、合成、旋转、模糊、叠加字幕等复杂的操作。其核心语法格式一般如下: ffmpeg -i input.mp4 -vf "滤镜参数" output.mp4或者带音频滤镜: ffmpeg…...

HBuilderX安装(uni-app和小程序开发)
下载HBuilderX 访问官方网站:https://www.dcloud.io/hbuilderx.html 根据您的操作系统选择合适版本: Windows版(推荐下载标准版) Windows系统安装步骤 运行安装程序: 双击下载的.exe安装文件 如果出现安全提示&…...

分布式增量爬虫实现方案
之前我们在讨论的是分布式爬虫如何实现增量爬取。增量爬虫的目标是只爬取新产生或发生变化的页面,避免重复抓取,以节省资源和时间。 在分布式环境下,增量爬虫的实现需要考虑多个爬虫节点之间的协调和去重。 另一种思路:将增量判…...
DeepSeek 技术赋能无人农场协同作业:用 AI 重构农田管理 “神经网”
目录 一、引言二、DeepSeek 技术大揭秘2.1 核心架构解析2.2 关键技术剖析 三、智能农业无人农场协同作业现状3.1 发展现状概述3.2 协同作业模式介绍 四、DeepSeek 的 “农场奇妙游”4.1 数据处理与分析4.2 作物生长监测与预测4.3 病虫害防治4.4 农机协同作业调度 五、实际案例大…...
深入理解Optional:处理空指针异常
1. 使用Optional处理可能为空的集合 在Java开发中,集合判空是一个常见但容易出错的场景。传统方式虽然可行,但存在一些潜在问题: // 传统判空方式 if (!CollectionUtils.isEmpty(userInfoList)) {for (UserInfo userInfo : userInfoList) {…...

【p2p、分布式,区块链笔记 MESH】Bluetooth蓝牙通信 BLE Mesh协议的拓扑结构 定向转发机制
目录 节点的功能承载层(GATT/Adv)局限性: 拓扑关系定向转发机制定向转发意义 CG 节点的功能 节点的功能由节点支持的特性和功能决定。所有节点都能够发送和接收网格消息。节点还可以选择支持一个或多个附加功能,如 Configuration …...

破解路内监管盲区:免布线低位视频桩重塑停车管理新标准
城市路内停车管理常因行道树遮挡、高位设备盲区等问题,导致车牌识别率低、逃费率高,传统模式在复杂路段束手无策。免布线低位视频桩凭借超低视角部署与智能算法,正成为破局关键。该设备安装于车位侧方0.5-0.7米高度,直接规避树枝遮…...

Xela矩阵三轴触觉传感器的工作原理解析与应用场景
Xela矩阵三轴触觉传感器通过先进技术模拟人类触觉感知,帮助设备实现精确的力测量与位移监测。其核心功能基于磁性三维力测量与空间位移测量,能够捕捉多维触觉信息。该传感器的设计不仅提升了触觉感知的精度,还为机器人、医疗设备和制造业的智…...