react useState的初始化函数 初始化值为props时的同步问题 | setState函数的使用与异步更新

本文主要是介绍react useState的初始化函数 初始化值为props时的同步问题 | setState函数的使用与异步更新,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • react setState函数
    • useState()钩子创建state
      • 如何根据props更新state值
    • setState的参数是下一个状态state
    • setState的参数是更新函数function
    • setState异步与同步
      • 合成事件
      • setState 实现原理

react setState函数

useState()钩子创建state

const [state, setState] = useState(initialState)`

useState钩子函数向组件添加一个状态变量,该函数返回状态变量(只用于显示)与设置状态变量的setState函数。

useState的使用细节

  1. 调用修改函数修改state变量的值(state值发生变化)会触发组件的重新渲染,直接修改state变量不会触发组件的重新渲染。
  2. 组件重新渲染等同于函数重新执行,函数体的代码会再次调用。执行useState时会返回本次渲染的state值。setState()是由钩子函数useState()生成的,useState()会保证数组的每次渲染都会获取到相同的setState()
  3. React只在初次渲染时保存初始状态,后续渲染将其忽略。

参数的两种形式

  1. 初始化函数
    惰性初始化函数:如果useState的参数是一个生成数据的函数,则可以直接将这个函数本身传递(仅在初始化期间调用一次),不是函数()(这种情况每次渲染时调用此函数)
    如果函数需要传参,可以在useState之前使用useMemo缓存该函数的结果。

  2. 参数是值 - 如果是props需要注意同步的问题

如何根据props更新state值

为什么父组件传递的props改变,子组件state值不同步更新?

父组件传递给子组件的props发生改变,引发子组件的render,并没有执行子组件的constructor函数(子组件没有被卸载自然不会重新加载)只会重新render。父组件的props传递给子组件的state,子组件的state只会在第一次加载的时候被赋值,后续的父组件props变化并不会被赋值到子组件的state

解决办法

  1. 子组件使用useEffect()
    useEffect的执行时机组件挂载->state改变->DOM改变->绘制屏幕->useEffect,所以如果在useEffect里更新state的数据,会在屏幕绘制成功后又重新开始这个流程。

    尽量不使用,官方解释:https://zh-hans.react.dev/learn/you-might-not-need-an-effect

    // 方法1:子组件使用useEffect
    const [tags, setTags] = useState(() => props.tags);useEffect(() => {
    setTags(props.tags);
    }, [props.tags]);
    
  2. 父组件使用子组件时,绑定key。当父组件要传递的props改变时改变key值,将子组件认为是不同的子组件(引发子组件重新加载而不是单纯的渲染) –推荐

    <children userId={userId} key={userId}/>
    

setState的参数是下一个状态state

  1. 在 React 中,状态被认为是只读的,因此应该替换它而不是改变现有对象。
  2. state值是一个对象时,setState(state)修改的参数,由于是下一次渲染的state值(组件重渲染后useState返回state值),需要使用新的对象去替换已有对象
// user是对象,对象的地址没有发生变化,所以不会引起组件重新渲染
const [user, setUser] = useState({name:"ranran",age:18})
user.name = "xxx"; 
setUser(user);// 解决方案:将其拷贝给另一个新对象,修改新对象的属性
setUser({...user,name:"xxx"}) // 后面的name会覆盖前面的name

setState()去修改一个state时,并不表示修改当前的state,修改的是组件下一次渲染的state(下一次 渲染中useState返回的内容)

说明案例1
假设name的初始值是Taylor,点击按钮后触发handleClick 函数,修改name的值,打印发现是修改前的值Taylor

// 点击按钮修改名字
const handleClick = ()=> {setName('Robin'); // 下一次渲染中useState返回的内容console.log(name); // Still "Taylor"! 本次渲染useState返回的内容
}

说明案例2
假设 age42,这个处理函数三次调用 setAge(age + 1),点击一次后,age 将只会变为 43

const handleClick() => {setAge(age + 1); // setAge(42 + 1)setAge(age + 1); // setAge(42 + 1)setAge(age + 1); // setAge(42 + 1)
}

setState的参数是更新函数function

更新函数的参数是待定状态state

const handleClick = () => {setAge(a => a + 1); // setAge(42 => 43)setAge(a => a + 1); // setAge(43 => 44)setAge(a => a + 1); // setAge(44 => 45)
}

React 将更新函数放入 队列 中。然后,在下一次渲染期间(组件重渲染,useState返回state的过程中),它将按照相同的顺序调用:
1.a => a + 1 将接收 42 作为待定状态,并返回 43 作为下一个状态。
2.a => a + 1 将接收 43 作为待定状态,并返回 44 作为下一个状态。
3.a => a + 1 将接收 44 作为待定状态,并返回 45 作为下一个状态。

按照惯例,通常将待定状态参数命名为状态变量名称的第一个字母,如 age 为 a。然而,你也可以把它命名为 prevAge 或者其他你觉得更清楚的名称。

批量更新
多个顺序的setState不是同步地一个一个执行,会一个一个加入队列,然后最后一起执行。在异步setState更新队列时,存储的是合并状态(上述案例age的最终状态45)。因此前面设置的 key 值会被后面所覆盖,最终只会执行一次更新。

setState异步与同步

明确这里所说的同步和异步指的是 API 调用后更新 DOM 是同步还是异步的。
react18之后已经全部异步了

设计为异步更新的原因

  1. setState设计为异步,可以显著的提升性能:如果每次调用setState都进行一次更新,那么意味着render函数会被频繁调用,界面重新渲染,这样效率是很低的;最好的办法应该是获取到多个更新,之后进行批量更新(这也为什么在react可控范围类,希望setState是异步的)。
  2. 如果同步更新了state,但是还没有执行render函数,而且props依赖于state中的数据,那么stateprops不能保持同步;stateprops不能保持一致性,会在开发中产生很多的问题。

setState是同步还是异步呢?

  • 在React内部机制能检测到的地方,组件生命周期(除componentDidUpdate) 或React合成事件中,setState是异步。
  • 如果 setState 在原生 JavaScript 控制的范围被调用,setState是同步。

setState 的“异步”并不是说内部由异步代码实现,其实本身执行的过程和代码都是同步的,只是合成事件和钩子函数的调用顺序在更新之前,导致在合成事件和钩子函数中没法立马拿到更新后的值,形成了所谓的“异步”。

合成事件

React 中为元素添加的事件被叫做合成事件(区分一下js的原生事件)

合成事件的好处

  • 屏蔽了浏览器之间关于事件处理的兼容性问题,为合成事件对象内部提供了统一的 API。
  • 提升性能, React 并不会将事件添加到真正的DOM 元素上,事件都被委托给 document

步骤说明
1.React 会在拥有事件的 DOM 对象身上添加一个 store 对象,在 store 对象中存储事件名称及事件处理函数,然后通过document 分发事件。
2.当事件被触发后,通过获取事件源对象,查看事件源对象中是否存在store对象,获取 store 对象中事件处理函数,执行事件处理函数。

核心实现
react 中所有的合成事件都会经过dispatchEventForLegacyPluginEventSystem()处理,其中通过设置全局变量isBatchingEventUpdates来标志当前的变化是否发生在React的可调度范围内。

/* 所有的事件都将经过此函数统一处理 */
function dispatchEventForLegacyPluginEventSystem(){// handleTopLevel 事件处理函数 batchedEventUpdates 批量更新batchedEventUpdates(handleTopLevel, bookKeeping);
}
function batchedEventUpdates(fn,a){/* 开启批量更新  */isBatchingEventUpdates = true;try {/* 这里执行了的事件处理函数, 比如在一次点击事件中触发setState,那么它将在这个函数内执行 */return batchedEventUpdatesImpl(fn, a, b);} finally {/* 完成一次事件,批量更新  */isBatchingEventUpdates = false;}
}

setState 实现原理

  1. 当 setState 方法被调用后,将状态传递给组件更新器,让组件更新器将状态临时存储起来。每个组件都会有自己的组件更新器,当需要更新组件时调用组件更新器。
  2. 状态临时保存完成后判断当前是否为批量更新模式,如果是,将组件更新器添加到更新队列中;如果不是,直接更新组件。
    当触发合成事件时, 在事件处理函数执行之前,会先将批量更新模式设置为 true,然后执行事件处理函数收集状态。当事件处理函数执行完成后,执行批量更新操作(从更新队列中获取组件更新器并调用)。组件更新器调用完成后再将批量更新模式设置为 false
  3. 更新组件时,先判断是否有状态需要更新,如果有就先计算最新状态,将得出的最新状态重新设置给组件(useState的返回值?)。
    计算状态时,如果状态是函数类型,调用函数传入当前状态,返回最新状态。如果状态是对象类型,使用对象状态覆盖原有状态。
  4. 组件状态计算完成后,通过调用组件内部的 render 方法获取新的 VirtualDOM,再通过 DOM 对象获取旧的虚拟 DOM,然后调用 diff 方法进行比对,对比完成后将差异更新到真实 DOM 对象中。

这篇关于react useState的初始化函数 初始化值为props时的同步问题 | setState函数的使用与异步更新的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/859576

相关文章

线上Java OOM问题定位与解决方案超详细解析

《线上JavaOOM问题定位与解决方案超详细解析》OOM是JVM抛出的错误,表示内存分配失败,:本文主要介绍线上JavaOOM问题定位与解决方案的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一、OOM问题核心认知1.1 OOM定义与技术定位1.2 OOM常见类型及技术特征二、OOM问题定位工具

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv

Spring Security简介、使用与最佳实践

《SpringSecurity简介、使用与最佳实践》SpringSecurity是一个能够为基于Spring的企业应用系统提供声明式的安全访问控制解决方案的安全框架,本文给大家介绍SpringSec... 目录一、如何理解 Spring Security?—— 核心思想二、如何在 Java 项目中使用?——

springboot中使用okhttp3的小结

《springboot中使用okhttp3的小结》OkHttp3是一个JavaHTTP客户端,可以处理各种请求类型,比如GET、POST、PUT等,并且支持高效的HTTP连接池、请求和响应缓存、以及异... 在 Spring Boot 项目中使用 OkHttp3 进行 HTTP 请求是一个高效且流行的方式。

Vue和React受控组件的区别小结

《Vue和React受控组件的区别小结》本文主要介绍了Vue和React受控组件的区别小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录背景React 的实现vue3 的实现写法一:直接修改事件参数写法二:通过ref引用 DOMVu

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解

《使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解》本文详细介绍了如何使用Python通过ncmdump工具批量将.ncm音频转换为.mp3的步骤,包括安装、配置ffmpeg环... 目录1. 前言2. 安装 ncmdump3. 实现 .ncm 转 .mp34. 执行过程5. 执行结

Java实现将HTML文件与字符串转换为图片

《Java实现将HTML文件与字符串转换为图片》在Java开发中,我们经常会遇到将HTML内容转换为图片的需求,本文小编就来和大家详细讲讲如何使用FreeSpire.DocforJava库来实现这一功... 目录前言核心实现:html 转图片完整代码场景 1:转换本地 HTML 文件为图片场景 2:转换 H

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法