React hooks文档笔记(三) 状态

news/2024/7/7 23:35:54

状态

  • 一、如何设计组件状态的步骤
  • 二、状态构造原则
    • 1. 组相关状态
    • 2. 避免矛盾/互斥状态
    • 3. 避免多余状态
    • 4. 不要把props放进state,除非你特别想要阻止更新
  • 三、状态保存/重置
    • 1. 相同位置的相同组件保留状态
    • 2. 同一位置不同元素reset状态

一、如何设计组件状态的步骤

  • 第一步:识别组件的不同视觉状态
  • 第二步:确定是什么触发了这些状态变化
  • 第三步:用useState代表状态
  • 第四步:移除任何非必要的状态变量
  • 第五步:连接事件处理程序以设置状态

二、状态构造原则

SSoT:单一数据源原则,

封装就是将强依赖的变量组织封装在一起

1. 组相关状态

如果同一时间需要更新多个状态,可以合并在单个状态中,如下:

const [x,setX]=useState(0)
const [y,setY]=useState(0)

⬇️

const [position, setPosition] = useState(
    { x: 0, y: 0 }
)

2. 避免矛盾/互斥状态

const [isSending, setIsSending] = useState(false)
const [isSent, setIsSent] = useState(false)

isSendingisSent不可能同时为true,所以最好用一个status状态来代替

⬇️

const [status, setStatus] = useState('sending'); //useState('sent')

3. 避免多余状态

遵循单一数据源原则,可以计算出的值不要再设为状态

const [firstName, setFirstName] = useState('');
const [lastName, setLastName] = useState('');
const [fullName, setFullName] = useState('');
function handleFirstNameChange(e) {
    setFirstName(e.target.value);
    setFullName(e.target.value + ' ' + lastName);
}
function handleLastNameChange(e) {
    setLastName(e.target.value);
    setFullName(firstName + ' ' + e.target.value);
}

fullName是一个计算值,可以根据firstNamelastName计算得出,不用设为状态

⬇️

const fullName = firstName + ' ' + lastName;

4. 不要把props放进state,除非你特别想要阻止更新

在父组件改变props.color时,以下代码不会重新渲染,状态仅在第一次渲染期间初始化。

export default function Clock(props) {
  const [color, setColor] = useState(props.color);
  return (
    <h1 style={{ color: color }}>
      {props.time}
    </h1>
  );
}

如果需要重新渲染 ⬇️

export default function Clock(props) {
  const color = props.color
  return (
    <h1 style={{ color: color }}>
      {props.time}
    </h1>
  );
}
  • 避免状态嵌套太深,需要重构数据结构

三、状态保存/重置

状态保存在 React 内部,React 通过组件在 UI 树中的位置将其持有的每个状态与正确的组件相关联。

1. 相同位置的相同组件保留状态

import { useState } from 'react';

// 当改变状态isPlayerA时,Counter因位置不变,保留之前的状态
export default function App() {
  const [isPlayerA, setIsPlayerA] = useState(true);
  return (
    <div>
      {isPlayerA ? (
        <Counter person="Taylor" />
      ) : (
        <Counter person="Sarah" />
      )}
    </div>
  )
}

两种重置状态的方法

  1. 在不同位置渲染组件
{isPlayerA &&
    <Counter person="Taylor" />
}
{!isPlayerA &&
    <Counter person="Sarah" />
}
  1. 给每个组件一个明确的标识key
{isPlayerA ? (
    <Counter person="Taylor" key="Taylor" />
) : (
    <Counter person="Sarah" key="Sarah" />
)}

2. 同一位置不同元素reset状态

(当您在同一位置渲染不同的组件时,它会重置其整个子树的状态)

当状态isFancy改变时,因为Counter组件被包裹的元素不同,整个子tree都会重新渲染,组件内部的状态会重置,

import { useState } from 'react';

export default function App() {
  const [isFancy, setIsFancy] = useState(false);
  return (
    <div>
      {isFancy ? (
        <div>
          <Counter isFancy={true} /> 
        </div>
      ) : (
        <section>
          <Counter isFancy={false} />
        </section>
      )}
    </div>
  )
}

ui tree变化如右图所示:在这里插入图片描述
演示如链接:
https://codesandbox.io/s/meowu5?file=%2FApp.js&utm_medium=sandpack

这就是为什么不应该嵌套组件函数,每点击一次按钮,输入状态就消失:
这是因为每次渲染都会创建不同的 函数,你在同一个位置渲染不同的组件,所以 React 重置下面的所有状态。
总是在顶层声明组件函数,不要嵌套它们的定义。

import { useState } from 'react';

export default function MyComponent() {
  const [counter, setCounter] = useState(0);

  function MyTextField() {
    const [text, setText] = useState('');

    return (
      <input
        value={text}
        onChange={e => setText(e.target.value)}
      />
    );
  }

  return (
    <>
      <MyTextField />
      <button onClick={() => {
        setCounter(counter + 1)
      }}>Clicked {counter} times</button>
    </>
  );
}

http://lihuaxi.xjx100.cn/news/1292431.html

相关文章

数据分析知识图谱

在做数据分析时&#xff0c;经常会有这样的困扰&#xff1a;面对几种相似的方法&#xff0c;既不清楚它们各自的使用场景&#xff0c;也无法分清它们之间的差别&#xff0c;一念之差就可能选错方法。如果你也有这样的困扰&#xff0c;建议按照SPSSAU知识图谱目录顺序检索对应的…

音频领域_DSP,专业词汇积累

serial &#xff1a;串行 parity&#xff1a;奇偶校验 handshake:握手 vendor:供应商 product:产品 …

【算法题】动态规划中级阶段之跳跃游戏、最大子数组和、解码方法

动态规划中级阶段 前言一、最大子数组和1.1、思路1.2、代码实现 二、跳跃游戏2.1、思路2.2、代码实现 三、解码方法3.1、思路3.2、代码实现 总结 前言 动态规划&#xff08;Dynamic Programming&#xff0c;简称 DP&#xff09;是一种解决多阶段决策过程最优化问题的方法。它是…

【业务功能篇35】Java Stream流 :Collectors.toMap

业务场景&#xff1a;我们经常会把一个list<Dto> 对象类集合&#xff0c;转换成一个map集合&#xff0c;例如一个问题单集合 list<proVo>, 我们需要转换成 一个map集合&#xff0c;key就指定 问题单类的某个字段&#xff0c;比如问题单号 no, value值就是对应的整…

libevent实践03:监听有名管道

首先创建两个管道&#xff1a;p1,p2 lkmaoubuntu:/big/libevent/libevent-2.1.12-stable/_install/learn$ mkfifo p1 p2 lkmaoubuntu:/big/libevent/libevent-2.1.12-stable/_install/learn$ ls -lshF p1 p2 0 prw-rw-r-- 1 lkmao lkmao 0 6月 29 20:39 p1| 0 prw-rw-r-- 1 l…

机器学习中的特征数据如何可视化

机器学习特征可视化是一种通过可视化方法来理解和解释机器学习模型中特征的重要性和影响力的技术。以下是一些常用的机器学习特征可视化技术的实现方法&#xff1a; 一&#xff09;直方图&#xff08;Histograms&#xff09; 直方图是一种常见的特征可视化方法&#xff0c;用于…

网页爬虫逆向与AST入门系列教程(八、AST的应用之安全检测)

网页爬虫逆向与AST入门系列教程 第八部分&#xff1a;AST的应用之安全检测 在前面的文章中&#xff0c;我们已经介绍了AST的基本概念、生成方法以及在代码混淆解析、反爬虫技术解析、数据提取与分析以及代码转换中的应用。在本篇中&#xff0c;我们将继续探讨AST在网页爬虫逆…

LLM - 搭建 ProteinGPT 结合蛋白质结构 PDB 知识的行业 ChatGPT 系统

欢迎关注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://blog.csdn.net/caroline_wendy/article/details/131403263 论文&#xff1a;ProteinChat: Towards Enabling ChatGPT-Like Capabilities on Protein 3D Structures 工程&#xff1a;ht…