zoukankan      html  css  js  c++  java
  • 关于 redux connect

    connect([mapStateToProps], [mapDispatchToProps], [mergeProps],[options])
    

    mapStateToProps

    mapStateToProps(state, ownProps) : stateProps
    

    这个函数允许我们将 store 中的数据作为 props 绑定到组件上。

    const mapStateToProps = (state) => {
      return {
        count: state.count
      }
    }
    

    (1)这个函数的第一个参数就是 Redux 的 store,我们从中摘取了 count 属性。你不必将 state 中的数据原封不动地传入组件,可以根据 state 中的数据,动态地输出组件需要的(最小)属性。

    (2)函数的第二个参数 ownProps,是组件自己的 props。有的时候,ownProps 也会对其产生影响。

    当 state 变化,或者 ownProps 变化的时候,mapStateToProps 都会被调用,计算出一个新的 stateProps,(在与 ownProps merge 后)更新给组件

    mapDispatchToProps

    connect 的第二个参数是 mapDispatchToProps,它的功能是,将 action 作为 props 绑定到组件上,也会成为 MyComp 的 props。

    [mergeProps],[options]

    不管是 stateProps 还是 dispatchProps,都需要和 ownProps merge 之后才会被赋给组件。connect 的第三个参数就是用来做这件事。通常情况下,你可以不传这个参数,connect 就会使用 Object.assign 替代该方法。

    [options] (Object) 如果指定这个参数,可以定制 connector 的行为。一般不用。

    原理解析

    首先connect之所以会成功,是因为Provider组件:

    • 在原应用组件上包裹一层,使原来整个应用成为Provider的子组件
    • 接收Redux的store作为props,通过context对象传递给子孙组件上的connect

    那connect做了些什么呢?

    它真正连接 Redux 和 React,它包在我们的容器组件的外一层,它接收上面 Provider 提供的 store 里面的 state 和 dispatch,传给一个构造函数,返回一个对象,以属性形式传给我们的容器组件。

    关于它的源码

    connect是一个高阶函数,首先传入mapStateToProps、mapDispatchToProps,然后返回一个生产Component的函数(wrapWithConnect),然后再将真正的Component作为参数传入wrapWithConnect,这样就生产出一个经过包裹的Connect组件,该组件具有如下特点:

    • 通过props.store获取祖先Component的store
    • props包括stateProps、dispatchProps、parentProps,合并在一起得到nextState,作为props传给真正的Component
    • componentDidMount时,添加事件this.store.subscribe(this.handleChange),实现页面交互
    • shouldComponentUpdate时判断是否有避免进行渲染,提升页面性能,并得到nextState
    • componentWillUnmount时移除注册的事件this.handleChange

    由于connect的源码过长,我们只看主要逻辑:

    export default function connect(mapStateToProps, mapDispatchToProps, mergeProps, options = {}) {
      return function wrapWithConnect(WrappedComponent) {
        class Connect extends Component {
          constructor(props, context) {
            // 从祖先Component处获得store
            this.store = props.store || context.store
            this.stateProps = computeStateProps(this.store, props)
            this.dispatchProps = computeDispatchProps(this.store, props)
            this.state = { storeState: null }
            // 对stateProps、dispatchProps、parentProps进行合并
            this.updateState()
          }
          shouldComponentUpdate(nextProps, nextState) {
            // 进行判断,当数据发生改变时,Component重新渲染
            if (propsChanged || mapStateProducedChange || dispatchPropsChanged) {
              this.updateState(nextProps)
                return true
              }
            }
            componentDidMount() {
              // 改变Component的state
              this.store.subscribe(() = {
                this.setState({
                  storeState: this.store.getState()
                })
              })
            }
            render() {
              // 生成包裹组件Connect
              return (
                <WrappedComponent {...this.nextState} />
              )
            }
          }
          Connect.contextTypes = {
            store: storeShape
          }
          return Connect;
        }
      }
    

    至此,上面的部分博客的都是从别人的文章中复制的
    原文:https://www.jianshu.com/p/4af924709b35

    connect 的优化实现:
    这是我自己的实现,现在 redux 并不是这样做的,高阶函数在 hooks 中并不需要

    const Connect = (mapStateToProps: (state: any) => any) => {
      return function wrapWithConnect(WrappedComponent: any) {
    
        const Container = (props: any) => {
          const [state, dispatch] = useContext(StoreContext)
          const storeProps = mapStateToProps(state)
          return <ConnectComponent {...storeProps} dispatch={dispatch} {...props}/>
        }
    
        const ConnectComponent: React.FC<any> = React.memo((props) => {
          return <WrappedComponent {...props} />
        }, (prevProps, nextProps) => {
          return _.isEqual(prevProps, nextProps)
        })
    
        return Container
      }
    }
    
    
    export default Connect
    

    Container 可以看做一个高阶函数, 他的主要作用就是比较新的 store 和旧 store 的参数是否发生变化,是否要阻止渲染

    如果使用 useContext, 只要在组件里面,那么就会 reRender, 没有 API 来阻止他的渲染(触发像我上面一样,放在高阶函数里面),不然的话只是用原生 react 的 API 就能完成所有 redux 的功能了

    demo 地址: https://github.com/Grewer/redux-connect-achieve

  • 相关阅读:
    SAXParseException;前言中不允许有内容的错误
    FATAL Alert:BAD_CERTIFICATE
    DB2的递归
    在Unity中针对屏幕自适应,我们该如何做呢?
    原码与反码的区别?
    在Unity 3D中加入Image图片
    你的外接键盘的小键盘在Num Lock键亮着的,但是数字按了不能用,解决办法在这里
    唯美英文(一)
    如何使用gcc编译器
    C++中const的用法
  • 原文地址:https://www.cnblogs.com/Grewer/p/12852465.html
Copyright © 2011-2022 走看看