React中状态设置this.setState()的实现

 更新时间:2025年09月18日 10:37:47   作者:歪歪100  
本文对比了React中类组件和函数组件的状态管理方式,前者合并更新,后者替换更新,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

React 中的状态设置,即 setState。这是一个核心概念,但它在 类组件和 函数组件(使用 Hook) 中有不同的写法和行为。

我会分两部分来解释。

第一部分:类组件中的setState

在类组件中,状态是一个叫 this.state 的对象,而更新状态的方法是 this.setState()

1. 基本用法

setState 用于更新组件的状态对象,并通知 React 需要重新渲染。

class Counter extends React.Component {
  constructor(props) {
    super(props);
    this.state = {
      count: 0
    };
  }

  increment() {
    // 传入一个新的对象来更新状态
    this.setState({ count: this.state.count + 1 });
  }

  render() {
    return (
      <div>
        <p>You clicked {this.state.count} times</p>
        <button onClick={() => this.increment()}>
          Click me
        </button>
      </div>
    );
  }
}

2. 关键特性:异步与批处理

setState 是异步的!
当你调用 setState 时,React 不会立即更新组件。它会将多个 setState 调用合并在一起(批处理 Batched Updates),然后进行一次更新以提高性能。

这意味着你不能在调用 setState 后立刻依赖 this.state 来获取最新值。

// ❌ 错误的做法
increment() {
  this.setState({ count: this.state.count + 1 });
  console.log(this.state.count); // 这里打印的仍然是旧的值,不是 +1 后的值
}

3. 如何依赖前一个状态?使用函数形式

为了解决异步问题并确保你基于最新的状态进行更新,setState 可以接受一个函数作为参数,而不是一个对象。

这个函数会接收先前的状态(prevState) 作为第一个参数。

increment() {
  this.setState((prevState) => {
    return { count: prevState.count + 1 }; // 基于之前的状态计算新状态
  });
}

这种形式在需要进行多次连续状态更新时尤其重要,因为它能保证每次更新都基于最新且正确的前一个状态。

4. 第二个参数:回调函数

setState 的第二个参数是一个可选的回调函数,它会在状态更新完成并且组件重新渲染后执行。你可以在这里执行一些依赖于新状态 DOM 的操作。

increment() {
  this.setState(
    (prevState) => ({ count: prevState.count + 1 }),
    () => {
      console.log('状态已更新,当前count是:', this.state.count); // 这里可以拿到最新值
      // 可以在这里操作DOM,比如根据新的状态设置焦点等
    }
  );
}

第二部分:函数组件中的setState(使用useStateHook)

在函数组件中,我们使用 useState Hook 来管理状态。它返回一个状态值和一个用于更新该状态的函数(通常命名为 setXxx)。

1. 基本用法

import { useState } from 'react';

function Counter() {
  // useState 返回一个数组: [当前状态值, 更新状态的函数]
  const [count, setCount] = useState(0);

  function increment() {
    // 直接传入新的值
    setCount(count + 1);
  }

  return (
    <div>
      <p>You clicked {count} times</p>
      <button onClick={increment}>
        Click me
      </button>
    </div>
  );
}

2. 同样具有异步和批处理特性

函数组件中的 setCount 也是异步的,并且也会被 React 批量处理。你不能在调用 setCount 后立刻读取 count 的值,因为它还没有被更新。

// ❌ 错误的做法
function increment() {
  setCount(count + 1);
  console.log(count); // 仍然是旧值
}

3. 如何依赖前一个状态?同样使用函数形式

与类组件类似,更新函数(如 setCount)也可以接受一个函数。这个函数接收先前的状态值作为其唯一参数。

function increment() {
  setCount((prevCount) => prevCount + 1);
}

这是推荐的做法,尤其是在事件循环中可能会多次更新同一个状态时,它能保证你总是在最新的状态基础上进行更新。

4. 与类组件setState的重大区别

特性类组件 this.setState函数组件 setXxx (Hook)
更新机制合并更新:传入的对象会与旧 state 浅合并。替换更新:直接用新值替换旧状态。不会自动合并对象!
示例this.setState({ user }); 只会更新 state.user,不影响 state.posts。setUser(newUser) 会直接用 newUser替换掉整个 user 状态。

函数组件中更新对象状态的正确做法:
因为你不能直接修改状态,必须创建一个新对象。

const [user, setUser] = useState({ name: 'Alice', age: 25 });

function updateName(newName) {
  // ❌ 错误:直接修改原对象
  // user.name = newName;
  // setUser(user); // 不会触发重新渲染,因为引用没变

  // ✅ 正确:展开运算符创建新对象
  setUser({
    ...user,      // 拷贝所有旧属性
    name: newName // 用新值覆盖其中的name属性
  });
}

总结与最佳实践

  1. 都是异步的:无论类组件还是函数组件,setState 都是异步操作,不要指望调用后能立刻拿到新值。
  2. 使用函数形式更新:当你新的状态依赖于旧的状态时(如计数器、列表追加等),务必使用 setXxx(prevState => newState) 的函数形式。这是最安全、最可靠的做法。
  3. 理解更新差异
    • 类组件:合并更新。
    • 函数组件:替换更新。更新对象时记得使用展开运算符 ...Object.assign() 来创建一个新对象。
  4. 副作用操作:在类组件中,使用 setState(updater, callback) 的第二个回调参数。在函数组件中,使用 useEffect Hook 来响应状态的变化。useEffect(() => { ... }, [count]);

到此这篇关于React中状态设置this.setState()的文章就介绍到这了,更多相关React 状态设置this.setState()内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家! 

相关文章

  • React实现随机颜色选择器的示例代码

    React实现随机颜色选择器的示例代码

    颜色选择器是一个用于选择和调整颜色的工具,它可以让用户选择他们喜欢的颜色,本文主要介绍了React实现随机颜色选择器的示例代码,具有一定的参考价值,感兴趣的可以了解一下
    2023-12-12
  • react 中 mobx的使用案例详解

    react 中 mobx的使用案例详解

    这篇文章主要介绍了react 中 mobx的使用案例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-04-04
  • React路由监听/跳转/守卫的完整指南

    React路由监听/跳转/守卫的完整指南

    React Router 是 React 单页应用的核心路由库,除了基础的路由配置,日常开发中还会高频用到路由监听、编程式跳转、路由守卫等进阶功能,本文从实战角度拆解这三大核心能力,需要的朋友可以参考下
    2026-03-03
  • react中hooks使用useState的更新不触发dom更新问题及解决

    react中hooks使用useState的更新不触发dom更新问题及解决

    这篇文章主要介绍了react中hooks使用useState的更新不触发dom更新问题及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-01-01
  • 基于react hooks,zarm组件库配置开发h5表单页面的实例代码

    基于react hooks,zarm组件库配置开发h5表单页面的实例代码

    这篇文章主要介绍了基于react hooks,zarm组件库配置开发h5表单页面,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-04-04
  • react中useEffect函数的详细用法(最新推荐)

    react中useEffect函数的详细用法(最新推荐)

    useEffect是React中的一个Hook,用于在函数组件中处理副作用(如数据获取、订阅、手动更改 DOM 等),useEffect属于组件的生命周期方法,下面通过本文给大家分享react中useEffect函数的详细用法,感兴趣的朋友跟随小编一起看看吧
    2024-06-06
  • React创建组件的三种方式及其区别

    React创建组件的三种方式及其区别

    本文主要介绍了React创建组件的三种方式及其区别,具有一定的参考价值,下面跟着小编一起来看下吧
    2017-01-01
  • redux工作原理讲解及使用方法

    redux工作原理讲解及使用方法

    这篇文章主要介绍了redux工作原理讲解及使用方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2021-11-11
  • React Native AsyncStorage本地存储工具类

    React Native AsyncStorage本地存储工具类

    这篇文章主要为大家分享了React Native AsyncStorage本地存储工具类,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-10-10
  • 在create-react-app中使用sass的方法示例

    在create-react-app中使用sass的方法示例

    这篇文章主要介绍了在create-react-app中使用sass的方法示例,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-10-10

最新评论