cari
Rumahhujung hadapan webhtml tutorialReact中跨组件分发状态的三种方法介绍
React中跨组件分发状态的三种方法介绍Aug 09, 2018 pm 01:35 PM
csshtmljavascriptreact.js

这篇文章给大家介绍的内容是关于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) =&gt; (
      <tr>
        <td>{book.title}</td>
        <td>
          &lt;button onClick={() =&gt; onEdit(index)}&gt;Edit&lt;/button&gt;
        </td>
      </tr>
    ))}
  </table>
);

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

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

render(&lt;BookApp /&gt;, document.getElementById("root"));

在codesandbox尝试一下

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

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

我们改如何修复它?

方法1:受控组件

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

//...

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

class BookApp extends Component {
  state = {
    books: books,
    activeBook: null,
    activeIndex: -1
  };
  render() {
    const { books, activeBook, activeIndex } = this.state;
    return (
      <div>
        &lt;BookList
          books={books}
          onEdit={index =&gt;
            this.setState({
              activeBook: { ...books[index] },
              activeIndex: index
            })}
        /&gt;
        &lt;BookForm
          book={activeBook}
          onChange={book =&gt; this.setState({ activeBook: book })}
          onSave={() =&gt;
            this.setState({
              books: Object.assign([...books], { [activeIndex]: activeBook }),
              activeBook: null,
              activeIndex: -1
            })}
        /&gt;
      </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 (
      &lt;form&gt;
        <h3>Book</h3>
        &lt;label&gt;
          Title:
          &lt;input
            value={this.state.title}
            onChange={e =&gt; this.setState({ title: e.target.value })}
          /&gt;
        &lt;/label&gt;
        &lt;label&gt;
          Author:
          &lt;input
            value={this.state.author}
            onChange={e =&gt; this.setState({ author: e.target.value })}
          /&gt;
        &lt;/label&gt;
        &lt;button onClick={() =&gt; this.props.onSave({ ...this.state })}&gt;
          Save
        &lt;/button&gt;
      &lt;/form&gt;
    );
  }
}
//...

在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>
    &lt;BookList
      books={books}
      onEdit={index =&gt;
        this.setState({
          activeIndex: index
        })}
    /&gt;
    &lt;BookForm
      key={activeIndex}
      book={activeBook}
      onSave={book =&gt;
        this.setState({
          books: Object.assign([...books], { [activeIndex]: book }),
          activeIndex: -1
        })}
    /&gt;
  </div>
);
}
}
//...

在codesandbox尝试一下。

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

相关文章推荐:

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

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

Atas ialah kandungan terperinci React中跨组件分发状态的三种方法介绍. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Kenyataan
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
HTML超文本标记语言--超在那里?(文档分析)HTML超文本标记语言--超在那里?(文档分析)Aug 02, 2022 pm 06:04 PM

本篇文章带大家了解一下HTML(超文本标记语言),介绍一下HTML的本质,HTML文档的结构、HTML文档的基本标签和图像标签、列表、表格标签、媒体元素、表单,希望对大家有所帮助!

React父组件怎么调用子组件的方法React父组件怎么调用子组件的方法Dec 27, 2022 pm 07:01 PM

调用方法:1、类组件中的调用可以利用React.createRef()、ref的函数式声明或props自定义onRef属性来实现;2、函数组件、Hook组件中的调用可以利用useImperativeHandle或forwardRef抛出子组件ref来实现。

web前端笔试题库之HTML篇web前端笔试题库之HTML篇Apr 21, 2022 am 11:56 AM

总结了一些web前端面试(笔试)题分享给大家,本篇文章就先给大家分享HTML部分的笔试题(附答案),大家可以自己做做,看看能答对几个!

怎么调试React源码?多种工具下的调试方法介绍怎么调试React源码?多种工具下的调试方法介绍Mar 31, 2023 pm 06:54 PM

怎么调试React源码?下面本篇文章带大家聊聊多种工具下的调试React源码的方法,介绍一下在贡献者、create-react-app、vite项目中如何debugger React的真实源码,希望对大家有所帮助!

深入理解React的自定义Hook深入理解React的自定义HookApr 20, 2023 pm 06:22 PM

React 自定义 Hook 是一种将组件逻辑封装在可重用函数中的方式,它们提供了一种在不编写类的情况下复用状态逻辑的方式。本文将详细介绍如何自定义封装 hook。

总结HTML中a标签的使用方法及跳转方式总结HTML中a标签的使用方法及跳转方式Aug 05, 2022 am 09:18 AM

本文给大家总结介绍a标签使用方法和跳转方式,希望对大家有所帮助!

react怎么设置div高度react怎么设置div高度Jan 06, 2023 am 10:19 AM

react设置div高度的方法:1、通过css方式实现div高度;2、在state中声明一个对象C,并在该对象中存放更换按钮的样式,然后获取A并重新设置C中的“marginTop”即可。

html中document是什么html中document是什么Jun 17, 2022 pm 04:18 PM

在html中,document是文档对象的意思,代表浏览器窗口的文档;document对象是window对象的子对象,所以可通过“window.document”属性对其进行访问,每个载入浏览器的HTML文档都会成为Document对象。

See all articles

Alat AI Hot

Undresser.AI Undress

Undresser.AI Undress

Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover

AI Clothes Remover

Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Undress AI Tool

Undress AI Tool

Gambar buka pakaian secara percuma

Clothoff.io

Clothoff.io

Penyingkiran pakaian AI

AI Hentai Generator

AI Hentai Generator

Menjana ai hentai secara percuma.

Artikel Panas

R.E.P.O. Kristal tenaga dijelaskan dan apa yang mereka lakukan (kristal kuning)
3 minggu yang laluBy尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Tetapan grafik terbaik
3 minggu yang laluBy尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Cara Memperbaiki Audio Jika anda tidak dapat mendengar sesiapa
3 minggu yang laluBy尊渡假赌尊渡假赌尊渡假赌

Alat panas

VSCode Windows 64-bit Muat Turun

VSCode Windows 64-bit Muat Turun

Editor IDE percuma dan berkuasa yang dilancarkan oleh Microsoft

SublimeText3 versi Mac

SublimeText3 versi Mac

Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

MantisBT

MantisBT

Mantis ialah alat pengesan kecacatan berasaskan web yang mudah digunakan yang direka untuk membantu dalam pengesanan kecacatan produk. Ia memerlukan PHP, MySQL dan pelayan web. Lihat perkhidmatan demo dan pengehosan kami.

Notepad++7.3.1

Notepad++7.3.1

Editor kod yang mudah digunakan dan percuma

Penyesuai Pelayan SAP NetWeaver untuk Eclipse

Penyesuai Pelayan SAP NetWeaver untuk Eclipse

Integrasikan Eclipse dengan pelayan aplikasi SAP NetWeaver.