使用 store 来优化 React 组件

前端之家收集整理的这篇文章主要介绍了使用 store 来优化 React 组件前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

在使用 React 编写组件的时候,我们常常会碰到两个不同的组件之间需要共享状态情况,而通常的做法就是提升状态到父组件。但是这样做会有一个问题,就是尽管只有两个组件需要这个状态,但是因为把状态提到了父组件,那么在状态变化的时候,父组件以及其下面的所有子组件都会重新 render,如果你的父组件比较复杂,包含了其他很多子组件的话,就有可能引起性能问题。

Redux 通过把状态放在全局的 store 里,然后组件去订阅各自需要的状态,当状态发生变化的时候,只有那些订阅的状态发生变化的组件才重新 render,这样就避免了上面说的提升状态所带来的副作用。但是,当我们在写一个 React 组件库的时候,redux 加 react-redux 的组合可能就有点太重了。所以我们可以自己写一个简单的 store,来实现类似 Redux 的订阅模式。

参考 Redux 的实现来写一个简版的 createStore:

  1. function createStore(initialState) {
  2. let state = initialState;
  3. const listeners = [];
  4.  
  5. function setState(partial) {
  6. state = {
  7. ...state,...partial,};
  8. for (let i = 0; i < listeners.length; i++) {
  9. listeners[i]();
  10. }
  11. }
  12.  
  13. function getState() {
  14. return state;
  15. }
  16.  
  17. function subscribe(listener) {
  18. listeners.push(listener);
  19.  
  20. return function unsubscribe() {
  21. const index = listeners.indexOf(listener);
  22. listeners.splice(index,1);
  23. };
  24. }
  25.  
  26. return {
  27. setState,getState,subscribe,};
  28. }

我们的 createStore 非常简单,算上空行也只有 33 行,总共暴露了 3 个方法,没有 Redux 里的 dispatch 和 reducer,直接通过 setState 方法改变状态。下面我们来用它一个计数器的例子(在线例子)。

  1. class Counter extends React.Component {
  2. constructor(props) {
  3. super(props);
  4.  
  5. // 初始化 store
  6. this.store = createStore({
  7. count: 0,});
  8. }
  9.  
  10. render() {
  11. return (
  12. <div>
  13. <Buttons store={store} />
  14. <Result store={store} />
  15. </div>
  16. )
  17. }
  18. }
  19.  
  20. class Buttons extends React.Component {
  21. handleClick = (step) => () => {
  22. const { store } = this.props;
  23. const { count } = store.getState();
  24. store.setState({ count: count + step });
  25. }
  26.  
  27. render() {
  28. return (
  29. <div>
  30. <button onClick={this.handleClick(1)}>+</button>
  31. <button onClick={this.handleClick(1)}>-</button>
  32. </div>
  33. );
  34. }
  35. }
  36.  
  37. class Result extends React.Component {
  38. constructor(props) {
  39. super(props);
  40.  
  41. this.state = {
  42. count: props.store.getState().count,};
  43. }
  44.  
  45. componentDidMount() {
  46. this.props.store.subscribe(() => {
  47. const { count } = this.props.store.getState();
  48. if (count !== this.state.count) {
  49. this.setState({ count });
  50. }
  51. });
  52. }
  53.  
  54. render() {
  55. return (
  56. <div>{this.state.count}</div>
  57. );
  58. };
  59. }

例子中 Buttons 里通过 store.setState 来改变 store 中的状态,并不会引起整个 Counter 的重新 render,但是因为 Result 中订阅了 store 的变化,所以当 count 有变化的时候就可以通过改变自己组件内的状态来重新 render,这样就巧妙地避免了不必须要的 render。

最后,上面的 createStore 虽然只有几十行代码,我还是把它写成了一个叫 mini-store 库放在 GitHub 上,并且提供了类似 Redux 的 Provider 和 connect 方法,总共加起来也就 100 多行代码。如果你也在写 React 组件库,需要管理一个复杂组件的状态,不妨试试这个优化方式。

猜你在找的React相关文章