精读React hook(一):useState 的几个基础用法和进阶技巧

自从 React v16.8 版本以来,React Hooks 为我们提供了全新的编写和思考 React 组件的方式。不仅管理状态和生命周期变得更简洁、更强大,Hooks 还引入了并发渲染和其他高级功能。本系列文章将详细探讨每一个 Hook,从基础到高级。首篇,我们将探讨最常用的 hook------useState

useState 的基本用法

useState主要用于给组件添加状态变量。注意,我们只能在组件的顶层或自定义的 Hooks 中调用useState

初始化状态

基础定义

tsx 复制代码
const [age, setAge] = useState(42);

懒初始化

对于需要计算得到的初始状态,可以使用函数传递给useState。这样,函数只在初次渲染时执行,而非每次渲染。

tsx 复制代码
const [todos, setTodos] = useState(createInitialTodos); // 注意:传递函数本身,非执行结果

更新状态

直接更新vs函数式更新

大部分情况,直接更新状态即可:

tsx 复制代码
setAge(newState);

但当新状态依赖于前一个状态时,推荐使用函数式更新。这确保了更新准确性,特别是在并发模式下。

tsx 复制代码
setState(prevState => prevState + 1);

以下两个例子展示函数式更新的重要性:

tsx 复制代码
// 示例1: 使用函数式更新
import { useState } from 'react';

export default function Counter() {
  const [age, setAge] = useState(42);

  function increment() {
    setAge(a => a + 1); // 函数式更新
  }

  return (
    <>
      <h1>Your age: {age}</h1>
      <button onClick={() => {
        increment();
        increment();
        increment();
      }}>+3</button>
    </>
  );
}
// 结果:点击 +3 时,age 更新为 45。
tsx 复制代码
// 示例2: 使用直接更新
import { useState } from 'react';

export default function Counter() {
  const [age, setAge] = useState(42);
  function increment() {
    setAge(age + 1); // 直接更新
  }
  return (
    <>
      <h1>Your age: {age}</h1>
      <button onClick={() => {
        increment();
        increment();
        increment();
      }}>+3</button>
    </>
  );
}
// 结果:点击 +3 时,可能只更新为 43。

对象与数组的更新

对象和数组的更新需要创建新的引用,而不是直接修改原状态。

tsx 复制代码
setForm({
  ...form,
  name: e.target.value // 更新这个属性
});

// 错误示例:
// form.name = e.target.value
tsx 复制代码
setTodos([
  ...todos,
  {
    id: nextId++,
    title: title,
    done: false
  }
]);

// 错误示例
// todos.push({
//   id: nextId++,
//   title: title,
//   done: false
// });
// setTodos(todos);

函数的更新

把函数存储到state里是很少见的做法,但某些情况下,我们有可能需要这么做。

先看一个错误的示例:

tsx 复制代码
const [fn, setFn] = useState(someFunction);

function handleClick() {
  setFn(someOtherFunction);
}

根据上文,我们知道这样的用法是把函数的返回值存储或更新到状态中,并不是把函数存储到状态中。

如果你想在状态中存储一个函数,你需要使用一个箭头函数来"包裹"它。这是正确的做法:

tsx 复制代码
const [fn, setFn] = useState(() => someFunction);

function handleClick() {
  setFn(() => someOtherFunction);
}

那么我们什么时候会需要这样使用?这里介绍一些可能的场景:

  1. 可配置的行为: 你可能有一个组件,它的行为可以由父组件进行配置。在这种情况下,你可以将函数作为状态存储,以便在组件的生命周期中更改或更新它。
  2. 动态创建的函数: 在某些情况下,你可能需要基于组件的某些属性或状态动态创建函数。将这些函数存储为状态可以确保你只在必要时重新创建它们。
  3. 回调和外部交互: 如果你的组件与外部系统交互,并且需要提供回调函数,你可能希望在状态中存储这些回调函数,以便在适当的时候更改或更新它们。
  4. 延迟执行的函数 : 在某些情况下,你可能想要在将来的某个时间点执行函数(例如,使用setTimeout)。将函数存储为状态可以确保即使组件的其他部分发生变化,你也可以访问到最初的函数引用。
  5. 与第三方库的集成: 有些第三方库可能要求你提供并在后续更改函数。在这种情况下,将函数作为状态存储可能会更加方便。

结语

在这篇文章中,我们深入探讨了 React 的useStateHook,从它的基础用法到一些进阶技巧。掌握好useState是走向 React 高手之路的关键一步。在未来的文章中,我们还将继续探讨其他的 Hooks。

相关推荐
Xf3n1an1 小时前
html语法
前端·html
张拭心1 小时前
亚马逊 AI IDE Kiro “狙击”Cursor?实测心得
前端·ai编程
烛阴1 小时前
为什么你的Python项目总是混乱?层级包构建全解析
前端·python
止观止2 小时前
React响应式组件范式:从类组件到Hooks
javascript·react.js·ecmascript
@大迁世界2 小时前
React 及其生态新闻 — 2025年6月
前端·javascript·react.js·前端框架·ecmascript
红尘散仙2 小时前
Rust 终端 UI 开发新玩法:用 Ratatui Kit 轻松打造高颜值 CLI
前端·后端·rust
新酱爱学习2 小时前
前端海报生成的几种方式:从 Canvas 到 Skyline
前端·javascript·微信小程序
袁煦丞3 小时前
把纸堆变数据流!Paperless-ngx让文件管理像打游戏一样爽:cpolar内网穿透实验室第539个成功挑战
前端·程序员·远程工作
慧慧吖@3 小时前
关于两种网络攻击方式XSS和CSRF
前端·xss·csrf
徐小夕3 小时前
失业半年,写了一款多维表格编辑器pxcharts
前端·react.js·架构