0

0

React中跨组件分发状态的三种方法介绍

不言

不言

发布时间:2018-08-09 13:35:13

|

1564人浏览过

|

来源于php中文网

原创

这篇文章给大家介绍的内容是关于react中跨组件分发状态的三种方法介绍,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。

当我问自己第一百次时,我正在研究一个典型的CRUD屏幕:“我应该将状态保留在这个组件中还是将其移动到父组件?”。

如果需要对子组件的状态进行轻微控制。您可能也遇到了同样的问题。

让我们通过一个简单的例子和三种修复方法来回顾它。前两种方法是常见的做法,第三种方法不太常规。

问题;

为了向您展示我的意思,我将使用一个简单的书籍CRUD(译者注:增加(Create)、读取查询(Retrieve)、更新(Update)和删除(Delete))屏幕(如此简单,它没有创建和删除操作)。

381280250-5b6ae9839a548_articlex.png

我们有三个组成部分。是一个组件,显示了用于编辑它们的书籍和按钮列表。有两个输入和一个按钮,用于保存对书籍的更改。以及包含其他两个组件的

那么,我们的状态是什么?好吧,应该跟踪书籍清单以及识别当前正在编辑的书籍的内容。 没有任何状态。并且应该保持输入的当前状态,直到单击“保存”按钮。

import React, { Component } from "react";
import { render } from "react-dom";

const books = [
  {
    title: "The End of Eternity",
    author: "Isaac Asimov"
  },
  //...
];

const BookList = ({ books, onEdit }) => (
  <table>
    <tr>
      <th>Book Title</th>
      <th>Actions</th>
    </tr>
    {books.map((book, index) => (
      <tr>
        <td>{book.title}</td>
        <td>
          <button onClick={() => onEdit(index)}>Edit</button>
        </td>
      </tr>
    ))}
  </table>
);

class BookForm extends Component {
  state = { ...this.props.book };
  render() {
    if (!this.props.book) return null;
    return (
      <form>
        <h3>Book</h3>
        <label>
          Title:
          <input
            value={this.state.title}
            onChange={e => this.setState({ title: e.target.value })}
          />
        </label>
        <label>
          Author:
          <input
            value={this.state.author}
            onChange={e => this.setState({ author: e.target.value })}
          />
        </label>
        <button onClick={() => this.props.onSave({ ...this.state })}>
          Save
        </button>
      </form>
    );
  }
}

class BookApp extends Component {
  state = {
    books: books,
    activeIndex: -1
  };
  render() {
    const { books, activeIndex } = this.state;
    const activeBook = books[activeIndex];
    return (
      <div>
        <BookList
          books={books}
          onEdit={index =>
            this.setState({
              activeIndex: index
            })}
        />
        <BookForm
          book={activeBook}
          onSave={book =>
            this.setState({
              books: Object.assign([...books], { [activeIndex]: book }),
              activeIndex: -1
            })}
        />
      </div>
    );
  }
}

render(<BookApp />, document.getElementById("root"));

在codesandbox尝试一下

看起来不错,但是他不起作用。

我们正在创建组件实例时初始化状态,因此,当从列表中选择另一本书时,父级无法让它知道它需要更改它。

我们改如何修复它?

方法1:受控组件

一种常见的方法是将状态提升,将转换为受控组件。我们删除状态,将activeBook添加到状态,并向添加一个onChange道具,我们在每次输入时都会调用它。

魅力企业网站管理系统2009 Sp6
魅力企业网站管理系统2009 Sp6

2009-8-24日更新1、全新升级网站自动升级通道,分为免费通道和商业通道!商业用户将获得更好技术支持服务。2、增加新闻中心,产品中心图片预览功能。操作更加直观,简单。3、支持系统集成第三方统计系统,各种在线客服系统4、魅力软件为了给用户提供更好的网站系统与服务,现特招募优秀开发人员和网页设计人员!系统介绍:一、无组件,开源。二、魅力软件坚持系统更新开发,不断完善。内置一键在线升级程序,方便您快

下载
//...

class BookForm extends Component {
  render() {
    if (!this.props.book) return null;
    return (
      <form>
        <h3>Book</h3>
        <label>
          Title:
          <input
            value={this.props.book.title}
            onChange={e =>
              this.props.onChange({
                ...this.props.book,
                title: e.target.value
              })}
          />
        </label>
        <label>
          Author:
          <input
            value={this.props.book.author}
            onChange={e =>
              this.props.onChange({
                ...this.props.book,
                author: e.target.value
              })}
          />
        </label>
        <button onClick={() => this.props.onSave()}>Save</button>
      </form>
    );
  }
}

class BookApp extends Component {
  state = {
    books: books,
    activeBook: null,
    activeIndex: -1
  };
  render() {
    const { books, activeBook, activeIndex } = this.state;
    return (
      <div>
        <BookList
          books={books}
          onEdit={index =>
            this.setState({
              activeBook: { ...books[index] },
              activeIndex: index
            })}
        />
        <BookForm
          book={activeBook}
          onChange={book => this.setState({ activeBook: book })}
          onSave={() =>
            this.setState({
              books: Object.assign([...books], { [activeIndex]: activeBook }),
              activeBook: null,
              activeIndex: -1
            })}
        />
      </div>
    );
  }
}

//...

方法2:同步state

现在它可以工作,但对我来说,提升的状态感觉不对。在用户单击“保存”之前,不关心对书的任何更改,那么为什么需要将其保持在自己的状态?

在codesandbox尝试一下

现在它可以工作,但对我来说,提升的状态感觉不对。在用户单击“保存”之前,不关心对书的任何更改,那么为什么需要将其保持在自己的状态?

//...
class BookForm extends Component {
  state = { ...this.props.book };
  componentWillReceiveProps(nextProps) {
    const nextBook = nextProps.book;
    if (this.props.book !== nextBook) {
      this.setState({ ...nextBook });
    }
  }
  render() {
    if (!this.props.book) return null;
    return (
      <form>
        <h3>Book</h3>
        <label>
          Title:
          <input
            value={this.state.title}
            onChange={e => this.setState({ title: e.target.value })}
          />
        </label>
        <label>
          Author:
          <input
            value={this.state.author}
            onChange={e => this.setState({ author: e.target.value })}
          />
        </label>
        <button onClick={() => this.props.onSave({ ...this.state })}>
          Save
        </button>
      </form>
    );
  }
}
//...

在codesandbox尝试一下

这种方法通常被认为是一种不好的做法,因为它违背了React关于拥有单一事实来源的想法。我不确定是这种情况,然而,同步状态并不总是那么容易。此外,我尽量避免使用生命周期方法。

方法3:由Key控制的组件

但为什么我们要回收旧的状态呢?每次用户选择一本书时,拥有一个全新状态的新实例是不是有意义?

为此,我们需要告诉React停止使用旧实例并创建一个新实例。这就是key prop的用途。

//...
class BookApp extends Component {
  state = {
books: books,
activeIndex: -1
};
  render() {
const { books, activeIndex } = this.state;
const activeBook = books[activeIndex];
return (
  <div>
    <BookList
      books={books}
      onEdit={index =>
        this.setState({
          activeIndex: index
        })}
    />
    <BookForm
      key={activeIndex}
      book={activeBook}
      onSave={book =>
        this.setState({
          books: Object.assign([...books], { [activeIndex]: book }),
          activeIndex: -1
        })}
    />
  </div>
);
}
}
//...

在codesandbox尝试一下。

如果元素具有与上一个渲染不同的键,则React会为其创建一个新实例。因此,当用户选择新书时,的键更改,将创建组件的新实例,并从props初始化状态。

相关文章推荐:

React Native跨域资源加载出错如何解决

React-JSX中如何实现Class与Style的动态绑定(附实例)

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

23

2026.03.06

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

68

2026.03.05

PHP高性能API设计与Laravel服务架构实践
PHP高性能API设计与Laravel服务架构实践

本专题围绕 PHP 在现代 Web 后端开发中的高性能实践展开,重点讲解基于 Laravel 框架构建可扩展 API 服务的核心方法。内容涵盖路由与中间件机制、服务容器与依赖注入、接口版本管理、缓存策略设计以及队列异步处理方案。同时结合高并发场景,深入分析性能瓶颈定位与优化思路,帮助开发者构建稳定、高效、易维护的 PHP 后端服务体系。

162

2026.03.04

AI安装教程大全
AI安装教程大全

2026最全AI工具安装教程专题:包含各版本AI绘图、AI视频、智能办公软件的本地化部署手册。全篇零基础友好,附带最新模型下载地址、一键安装脚本及常见报错修复方案。每日更新,收藏这一篇就够了,让AI安装不再报错!

84

2026.03.04

Swift iOS架构设计与MVVM模式实战
Swift iOS架构设计与MVVM模式实战

本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

113

2026.03.03

C++高性能网络编程与Reactor模型实践
C++高性能网络编程与Reactor模型实践

本专题围绕 C++ 在高性能网络服务开发中的应用展开,深入讲解 Socket 编程、多路复用机制、Reactor 模型设计原理以及线程池协作策略。内容涵盖 epoll 实现机制、内存管理优化、连接管理策略与高并发场景下的性能调优方法。通过构建高并发网络服务器实战案例,帮助开发者掌握 C++ 在底层系统与网络通信领域的核心技术。

29

2026.03.03

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

79

2026.02.28

Golang 工程化架构设计:可维护与可演进系统构建
Golang 工程化架构设计:可维护与可演进系统构建

Go语言工程化架构设计专注于构建高可维护性、可演进的企业级系统。本专题深入探讨Go项目的目录结构设计、模块划分、依赖管理等核心架构原则,涵盖微服务架构、领域驱动设计(DDD)在Go中的实践应用。通过实战案例解析接口抽象、错误处理、配置管理、日志监控等关键工程化技术,帮助开发者掌握构建稳定、可扩展Go应用的最佳实践方法。

62

2026.02.28

Golang 性能分析与运行时机制:构建高性能程序
Golang 性能分析与运行时机制:构建高性能程序

Go语言以其高效的并发模型和优异的性能表现广泛应用于高并发、高性能场景。其运行时机制包括 Goroutine 调度、内存管理、垃圾回收等方面,深入理解这些机制有助于编写更高效稳定的程序。本专题将系统讲解 Golang 的性能分析工具使用、常见性能瓶颈定位及优化策略,并结合实际案例剖析 Go 程序的运行时行为,帮助开发者掌握构建高性能应用的关键技能。

51

2026.02.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React.JS中文基础视频教程
React.JS中文基础视频教程

共14课时 | 3万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3.5万人学习

AngularJS教程
AngularJS教程

共24课时 | 4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号