在 React 中过滤表的最佳方法

Best way to filter table in React

本文关键字:最佳 方法 过滤 React      更新时间:2023-09-26

我有一个存储在redux中的对象数组。我希望能够根据用户输入过滤该数组。我应该创建一个通过 props 接收数组的状态对象并修改该数组,还是混合状态和 prop 是一种不好的做法?如果可以混合两者,我应该在componentWillReceiveProps中设置状态吗?

基于道具构建状态可能有些复杂,这是可以接受的,但您应该考虑所有选项。

最简单的实现是过滤render方法中的 props。如果您有足够小的组件由于太多原因而没有更新,特别是如果列表中的元素数量很少,这可能是首选方法:

class FilterList extends React.Component {
  render () {
    const { elements } = this.props;
    const { filterStr } = this.state;
    const filteredElements = elements
      .filter(e => e.includes(filterStr))
      .map(e => <li>{ e }</li>)
    return (
      <div>
        <input
          type="text"
          value={ filterStr }
          onChange={ e => this.setState({ filterStr: e.target.value }) } />
        <ul>
          { filteredElements }
        </ul>
      </div>
    );
  }
}

下一个选项是执行您所描述的操作,并根据组件的过滤器状态和传递给它的 props 派生计算状态。当您有一个复杂的组件接收许多道具并且经常渲染时,这很好。在这里,您缓存可查看的元素,并且仅在需要过滤时才过滤列表。

class FilterList extends React.Component {
  constructor (props) {
    this.state = {
      viewableEls: props.elements
    }
  }
  componentWillReceiveProps (nextProps) {
    const { elements } = this.props;
    const { filterStr } = this.state;
    if (elements !== nextProps.elements) {
      this.setState({
        viewableEls: this.getViewableEls(nextProps.elements, filterStr)
      })
    }
  }
  getViewableEls (elements, filterStr) {
    return elements.filter(el => el.includes(filterStr))
  }
  handleFilterChange = e => {
    const { elements } = this.props;
    this.setState({
      filterStr: e.target.value,
      viewableEls: this.getViewableEls(elements, filterStr)
    })
  }
  render () {
    const { viewableEls } = this.state;
    return (
      <div>
        <input
          type="text"
          value={ filterStr }
          onChange={ this.handleFilterChange } />
        <ul>
          { viewableEls.map(e => <li key={ e }>{ e }</li>) }
        </ul>
      </div>
    );
  }
}

最后,redux 'way',它要求你将动作创建者和filterStr作为道具传递给组件,可能通过connect其他地方传入。下面的实现使用的是无状态组件,因为我们根本不会将fitlerStr保持在组件状态。

const FilterTable = ({ elements, filterStr, changeFilterStr }) => {
  return (
    <div>
      <input
        type="text"
        value={ filterStr }
        onChange={ e => changeFilterStr(e.target.value) } />
      <ul>
        {
          elements
            .filter(e => e.includes(filterStr))
            .map(e => <li key={ e }>{ e }</li>)
        }
      </ul>
    </div>
  )
}