首页 > 解决方案 > React 卸载子组件的方式是什么?

问题描述

我是 React 的新手,但是当我学习 componentWillUnmount() 方法时,我发现了一种我无法理解的行为。

考虑这个脚本(这里是小提琴-> https://jsfiddle.net/ockhamsblade/4zf5tq2h/37/):

let myArray = [1,2,3,4,5,6];

class List extends React.Component {

  constructor() {
    super()
    this.state = {
      list: []
    };
  }

  componentDidMount() {
    this.setState({list:myArray});
  }
    render() {
    return (
    <div>
      <h1>My list</h1>
        <ul>
        {
            this.state.list.map((item, index) => (
            <Item key={index} value={item} onDeleteItem={this.handleDelete}>Item {item}</Item>
          ))
            }
          </ul>
    </div>
    )
  }

  handleDelete = (value) => {
    console.log("Original list: ", this.state.list);
    console.log("Handling delete of item " + value);
    let newState = {...this.state};
    newState.list = newState.list.filter((v) => v !== value);
    console.log("New list: ", newState.list);
    this.setState(newState);
  }
}

class Item extends React.Component {
    render() {
  return (<li><button type="button" onClick={this.delete}>Item {this.props.value}</button></li>);
  }

  delete = () => {
    console.log('Please remove item ' + this.props.value);
    this.props.onDeleteItem(this.props.value);
  }

  componentWillUnmount() {
    console.log("Unmounting item " + this.props.value);
  }
}

ReactDOM.render(<List />, document.querySelector("#app"))

正如您可以检查 JSFiddle.net 脚本(使用开发人员的控制台)一样,每当您单击一个项目时,List 组件都会从其状态中删除该值并重新呈现剩余的项目。到现在为止还挺好。问题是,在卸载之前,Item 组件始终具有最后一个 item 的 props,例如,如果这些将被覆盖或其他情况(您可以在开发人员的控制台中查看这一点)。所以,即使我要删除项目#3(或任何一个),componentWillUnmount() 上的“this.props.value”总是 = 6,而我期望 = 3。

我猜想,无论出于何种性能原因,React 都会从 DOM 中删除最后一项并重新渲染其余项。这种行为是预期的吗?如果是这样,您能否参考一些文档和/或告诉我如何在卸载之前保留该项目的原始实例?我试图将 props.value 存储在项目状态上,但也不起作用。

提前致谢。

标签: javascriptreactjsreact-state-managementreact-component-unmount

解决方案


你得到这个的原因是因为你将数组的索引作为键传递。相反,如果您传递一个有效的唯一标识符,则不会发生这种情况。

请运行此代码,您将理解。

let myArray = [1,2,3,4,5,6];

class List extends React.Component {

  constructor() {
    super()
    this.state = {
      list: []
    };
  }

  componentDidMount() {
    this.setState({list:myArray});
  }
    render() {
    return (
    <div>
      <h1>My list</h1>
        <ul>
        {
            this.state.list.map((item, index) => (
            <Item key={`${item}`} value={item} onDeleteItem={this.handleDelete}>Item {item}</Item>
          ))
            }
          </ul>
    </div>
    )
  }

  handleDelete = (value) => {
    console.log("Original list: ", this.state.list);
    console.log("Handling delete of item " + value);
    let newState = {...this.state};
    newState.list = newState.list.filter((v) => v !== value);
    console.log("New list: ", newState.list);
    this.setState(newState);
  }
}

class Item extends React.Component {
    render() {
  return (<li><button type="button" onClick={this.delete}>Item {this.props.value}</button></li>);
  }

  delete = () => {
    console.log('Please remove item ' + this.props.value);
    this.props.onDeleteItem(this.props.value);
  }
  componentDidUpdate(){
    console.log("incdu",this.props.value);
  }

  componentWillUnmount() {
    console.log("Unmounting item " + this.props.value);
  }
}

ReactDOM.render(<List />, document.querySelector("#app"))

我在这里所做的唯一更改是将正确的标识符(这里是项目本身)作为键传递。

<Item key={`${item}`} value={item} onDeleteItem={this.handleDelete}>Item {item}</Item>

你可以参考Understanding unique keys for array children in React.js


推荐阅读