React的react-side-effect/react-document.title源码浅析

问题1:基本概念

(1)创建一个组件,该组件的prop的改变会被映射到全局的side effect上(因为如果一个组件的props改变,那么该数组中放的所有的组件实例都会调用emitChange,并根据他们的props得到我们最终的state)
(2)和componentDidMount的不同
他会收集所有当前整棵树的props才会把他传递给side effect(我的理解是handleStateChangeOnClient/mapStateOnServer方法

(3)API签名:

  withSideEffect: (reducePropsToState,handleStateChangeOnClient,[mapStateOnServer]) -> ReactComponent -> ReactComponent

这是一个高阶组件,当mounting,unmounting or receiving new props等生命周期方法调用的时候都会调用reducePropsToState,同时传入每一个已经挂载的实例对象的props,你可以操作这些挂载的props然后返回一个state。在客户端,每次这个组件被挂载/卸载或者他的props属性发生变化,reducePropsToState都会被调用,同时这个重新计算得到的state对象会被传入到handleStateChangeOnClient,在这函数中你可以使用这个state去触发side effect;在服务端handleStateChangeOnClient不会被触发,但是在调用renderToString后你可以调用返回的组件的静态方法rewind去获取当前的state,如果在调用了renderToString后你忘记了调用rewind,那么内部的组件实例对象的调用栈会继续增长,最后会产生内存泄漏和错误的信息,因此在服务端每次调用renderToString后你必须手动调用rewind;在测试环境中,你可以使用返回的组件的静态方法peek,他允许你获取当前的state同时不会重置已经挂载的组件的实例栈,但是在非测试环境中不要使用
(4)调用方式如下,其会返回一个新的组件实例SideEffect

  export default withSideEffect(
        reducePropsToState,//会找到所有的实例对象的props,并得到最终的state
        handleStateChangeOnClient//
      )(BodyStyle);

问题2:源码如下
module.exports = function withSideEffect(
  reducePropsToState,mapStateOnServer
) {
  if (typeof reducePropsToState !== 'function') {
    throw new Error('Expected reducePropsToState to be a function.');
  }
  if (typeof handleStateChangeOnClient !== 'function') {
    throw new Error('Expected handleStateChangeOnClient to be a function.');
  }
  if (typeof mapStateOnServer !== 'undefined' && typeof mapStateOnServer !== 'function') {
   throw new Error('Expected mapStateOnServer to either be undefined or a function.');
  }

  //包裹元素的displayName,name,如果没有这两个属性,那么返回'Component'字符串
  function getdisplayName(WrappedComponent) {
    return WrappedComponent.displayName || WrappedComponent.name || 'Component';
  }

  //接受一个包裹组件WrappedComponent,同时返回一个组件,也就是SideEffect组件的实例对象
  return function wrap(WrappedComponent) {
    if (typeof WrappedComponent !== 'function') {
      throw new Error('Expected WrappedComponent to be a React component.');
    }

    let mountedInstances = [];
    //已经挂载的组件的实例

    let state;

    //如果组件已经更新,那么对每一个组件实例都调用我们的reducePropsToState方法并得到最终的state
    //如果可以操作DOM,那么我们调用handleStateChangeOnClient方法并传入我们的最终的state,否则
    //调用mapStateOnServer并传入我们的state
    function emitChange() {
      state = reducePropsToState(mountedInstances.map(function (instance) {
        return instance.props;
      }));

      if (SideEffect.canUseDOM) {
        handleStateChangeOnClient(state);
      } else if (mapStateOnServer) {
        state = mapStateOnServer(state);
      }
    }

    //这个类实际上是一个函数,其可以访问我们外部定义的mountedInstances数组,这样每个WrappedComponent实例
    //都是有一个自己的私有的实例数组的!
    class SideEffect extends Component {
      // Try to use displayName of wrapped component
      static displayName = `SideEffect(${getdisplayName(WrappedComponent)})`;
      //静态属性

      // Expose canUseDOM so tests can monkeypatch it
      static canUseDOM = ExecutionEnvironment.canUseDOM;
      //是否可以使用DOM

      //获取当前的state,同时不会重置当前已经挂载的组件实例栈,不要在非测试环境中使用
      static peek() {
        return state;
      }

      //在服务端handleStateChangeOnClient不会被触发,但是在调用renderToString后你可以调用返回的组件的静态方法rewind去获取当前的state,如果在调用了renderToString后你忘记了调用rewind,那么内部的组件实例对象的调用栈会
      //继续增长,最后会产生内存泄漏和错误的信息,因此在服务端每次调用renderToString后你必须手动调用rewind
      static rewind() {
        //服务端才能调用
        if (SideEffect.canUseDOM) {
          throw new Error('You may only call rewind() on the server. Call peek() to read the current state.');
        }
        let recordedState = state;
        state = undefined;
        mountedInstances = [];
        //返回state,但是已经挂载的实例会全部被重置,同时state被重置为null
        return recordedState;
      }
      
      //接受SideEffect的新的props属性,如果属性值不相同那么才会更新组件
      shouldComponentUpdate(nextProps) {
        return !shallowEqual(nextProps,this.props);
      }

      //你每次实例化SideEffect这个类的时候,就会在数组中放入我们的this对象并遍历所有的实例对象得到最终的state
      componentwillMount() {
        mountedInstances.push(this);
        emitChange();
      }
    
      //如果组件已经更新
      componentDidUpdate() {
        emitChange();
      }
      
      //找到卸载的这个组件在数组中的下标并删除,同时调用emitChange
      componentwillUnmount() {
        const index = mountedInstances.indexOf(this);
        mountedInstances.splice(index,1);
        emitChange();
      }

      //此处的SideEffect实例化传入的props会被传入到我们的包裹组件WrappedComponent中,此处就是高阶组件
      //每一个实例对象都会调用一次render方法
      render() {
        return <WrappedComponent {...this.props} />;
      }
    }

    return SideEffect;
  }
}

问题3:在react-document-title中的使用

'use strict';

var React = require('react'),withSideEffect = require('react-side-effect');

//传入每一个被挂载的组件的props属性,然后让开发者自己控制应该返回的state属性
function reducePropsToState(propsList) {
  var innermostProps = propsList[propsList.length - 1];
  if (innermostProps) {
    return innermostProps.title;
  }
}

//重新计算得到的state会被传入到这个函数
function handleStateChangeOnClient(title) {
  var nextTitle = title || '';
  if (nextTitle !== document.title) {
    document.title = nextTitle;
  }
}

//接受一个DocumentTitle组件,返回一个SideEffect实例
var DocumentTitle = React.createClass({
  displayName: 'DocumentTitle',propTypes: {
    title: React.PropTypes.string.isrequired
  },//每次实例化一个DocumentTitle都会调用该render方法
  render: function render() {
    if (this.props.children) {
      return React.Children.only(this.props.children);
    } else {
      return null;
    }
  }
});

//调用方式如下:
/*
 <DocumentTitle title='Home'>
    <h1>Home,sweet home.</h1>
  </DocumentTitle>
*/ 
module.exports = withSideEffect(
  reducePropsToState,handleStateChangeOnClient
)(DocumentTitle);

参考资料:

React-side-effect

react-document-title

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


react 中的高阶组件主要是对于 hooks 之前的类组件来说的,如果组件之中有复用的代码,需要重新创建一个父类,父类中存储公共代码,返回子类,同时把公用属性...
我们上一节了解了组件的更新机制,但是只是停留在表层上,例如我们的 setState 函数式同步执行的,我们的事件处理直接绑定在了 dom 元素上,这些都跟 re...
我们上一节了解了 react 的虚拟 dom 的格式,如何把虚拟 dom 转为真实 dom 进行挂载。其实函数是组件和类组件也是在这个基础上包裹了一层,一个是调...
react 本身提供了克隆组件的方法,但是平时开发中可能很少使用,可能是不了解。我公司的项目就没有使用,但是在很多三方库中都有使用。本小节我们来学习下如果使用该...
mobx 是一个简单可扩展的状态管理库,中文官网链接。小编在接触 react 就一直使用 mobx 库,上手简单不复杂。
我们在平常的开发中不可避免的会有很多列表渲染逻辑,在 pc 端可以使用分页进行渲染数限制,在移动端可以使用下拉加载更多。但是对于大量的列表渲染,特别像有实时数据...
本小节开始前,我们先答复下一个同学的问题。上一小节发布后,有小伙伴后台来信问到:‘小编你只讲了类组件中怎么使用 ref,那在函数式组件中怎么使用呢?’。确实我们...
上一小节我们了解了固定高度的滚动列表实现,因为是固定高度所以容器总高度和每个元素的 size、offset 很容易得到,这种场景也适合我们常见的大部分场景,例如...
上一小节我们处理了 setState 的批量更新机制,但是我们有两个遗漏点,一个是源码中的 setState 可以传入函数,同时 setState 可以传入第二...
我们知道 react 进行页面渲染或者刷新的时候,会从根节点到子节点全部执行一遍,即使子组件中没有状态的改变,也会执行。这就造成了性能不必要的浪费。之前我们了解...
在平时工作中的某些场景下,你可能想在整个组件树中传递数据,但却不想手动地通过 props 属性在每一层传递属性,contextAPI 应用而生。
楼主最近入职新单位了,恰好新单位使用的技术栈是 react,因为之前一直进行的是 vue2/vue3 和小程序开发,对于这些技术栈实现机制也有一些了解,最少面试...
我们上一节了了解了函数式组件和类组件的处理方式,本质就是处理基于 babel 处理后的 type 类型,最后还是要处理虚拟 dom。本小节我们学习下组件的更新机...
前面几节我们学习了解了 react 的渲染机制和生命周期,本节我们正式进入基本面试必考的核心地带 -- diff 算法,了解如何优化和复用 dom 操作的,还有...
我们在之前已经学习过 react 生命周期,但是在 16 版本中 will 类的生命周期进行了废除,虽然依然可以用,但是需要加上 UNSAFE 开头,表示是不安...
上一小节我们学习了 react 中类组件的优化方式,对于 hooks 为主流的函数式编程,react 也提供了优化方式 memo 方法,本小节我们来了解下它的用...
开源不易,感谢你的支持,❤ star me if you like concent ^_^
hel-micro,模块联邦sdk化,免构建、热更新、工具链无关的微模块方案 ,欢迎关注与了解
本文主题围绕concent的setup和react的五把钩子来展开,既然提到了setup就离不开composition api这个关键词,准确的说setup是由...
ReactsetState的执行是异步还是同步官方文档是这么说的setState()doesnotalwaysimmediatelyupdatethecomponent.Itmaybatchordefertheupdateuntillater.Thismakesreadingthis.staterightaftercallingsetState()apotentialpitfall.Instead,usecom