React 模式,技巧,技巧和窍门

- React FrontEnd
✨✨✨You can Edit this Article on Github site
✏️✏️✏️ This article .MD file was last updated at: Loading ...

这篇文章整理了学习 React 过程中以及实际开发应用当中一些模式,技巧,技巧和窍门。

大部门内容是基于 React 框架下面产生的一些内容,有很大的局限性,但是确实带来了新的理念和开发方式,仁者见仁智者见智,多学习一点内容总归对职业生涯有好处。

本篇文章还在持续更新中,如果有错误烦请指正。

Normally React

一些常见的关于 React 需要了解的内容,就简单列举如下。下面只会列到本人认为比较值得重视的部分进行详细陈述。

语法层面——基础入门

  • JSX 语法、React 基本内容等
  • Derocator 或者 async await 等常见 ES6、ES7 的内容
  • React Lists and Keys

常用层面——日常开发必备

进阶层面——性能优化和业务解耦必备

  • PureRenderMixin 以及 Pure Component 以及 shouldComponentUpdate check
  • unstable_rendersubtreeintocontainer 以及 portal
  • React fiber 更新机制
  • React call return

虚拟 DOM

一些周边拓展(含有本人观念

  • styled components
  • Redux saga
  • Recompose

最后强烈建议推荐这个:List of top 301 ReactJS Interview Questions & Answers

Pure Render Checks

// 坏的例子
class Table extends PureComponent {

  update(e) {
    this.props.update(e.target.value);
  }

  render() {
    return (
      <div>
        {this.props.items.map(i =>
          <Cell
            data={i}
            options={this.props.options || []}
            onChange={this.update.bind(this)}
            onClick={e => this.props.update(e.target.value)}/>;
           />
        )}
      </div>
    );
  }
}

这种写法的问题在于{this.props.options || []} 这种写法会导致所有的 Cell 都被重新渲染即使只有一个 cell 发生改变。

原因是:每次传入的[]都相当于创建了新的 Array 实例。在 JavaScript 里面,不同的实例是有不同的实体的,所以浅比较在这种情况下总是会返回 false,然后组件就会被重新渲染。

对于onChangeonClick也是一样的问题,也会导致重新渲染。

// 好的例子
const defaultval = [] // <---  也可以使用defaultProps
class Table extends PureComponent {
  update(e) {
    this.props.update(e.target.value)
  }

  render() {
    return (
      <div>
        {this.props.items.map(i => (
          <Cell
            data={i}
            options={this.props.options || defaultval}
            onChange={this.update}
            onClick={this.update}
          />
        ))}
      </div>
    )
  }
}

总结就是: props 上的任何 prop 值尽量只创建一次, 只绑定一次

Reference:

Synthetic events in React

React 在处理事件(event 时), 事实上使用了 SyntheticEvent 对象包裹了原生的 event 对象.

这些 React 自己维护的对象是相互联系的, 意味着如果对于某一个事件, 我们给出了对应的响应函数(handler), 其他的 SyntheticEvent 对象也是可以重用的.这也是 React 提升性能的秘诀之一. 但是这也意味着, 如果想要通过异步的方式访问事件对象是不可能的, 因为出于 reuse 的原因, 事件对象里面的值都被重置了.

下面这段代码会在控制台里面打出 null, 因为事件在 SyntheticEvent 池中被重用了.

function handleClick(event) {
  setTimeout(function () {
    console.log(event.target.name);
  }, 1000);
}

为了避免这种情况, 你需要去保存你关心的事件的属性.

function handleClick(event) {
  let name = event.target.name;
  setTimeout(function () {
    console.log(name);
  }, 1000);
}

Reference:

async-nature-of-setState

在某些情况下,React 框架出于性能优化考虑,可能会将多次 state 更新合并成一次更新。正因为如此,setState 实际上是一个异步的函数。 但是,有一些行为也会阻止 React 框架本身对于多次 state 更新的合并,从而让 state 的更新变得同步化。 比如: eventListeners, Ajax, setTimeout 等等。

详解

当 setState() 函数执行的时候,函数会创建一个暂态的 state 作为过渡 state,而不是立即修改 this.state。 如果在调用 setState() 函数之后尝试去访问 this.state,你得到的可能还是 setState() 函数执行之前的结果。 在使用 setState() 的情况下,看起来同步执行的代码其实执行顺序是得不到保证的。原因上面也提到过,React 可能会将多次 state 更新合并成一次更新来优化性能。

运行下面这段代码,你会发现当和 addEventListener , setTimeout 函数或者发出 AJAX call 的时候,调用 setState , state 会发生改变。并且 render 函数会在 setState() 函数被触发之后马上被调用。那么到底发生了什么呢?事实上,类似 setTimeout() 函数或者发出 ajax call 的 fetch 函数属于调用浏览器层面的 API,这些函数的执行并不存在与 React 的上下文中,所以 React 并不能够像控制其他存在与其上下文中的函数一样,将多次 state 更新合并成一次。

在上面这些例子中,React 框架之所以在选择在调用 setState 函数之后立即更新 state 而不是采用框架默认的方式,即合并多次 state 更新为一次更新,是因为这些函数调用(fetch,setTimeout 等浏览器层面的 API 调用)并不处于 React 框架的上下文中,React 没有办法对其进行控制。React 在此时采用的策略就是及时更新,确保在这些函数执行之后的其他代码能拿到正确的数据(即更新过的 state)。

class TestComponent extends React.Component {

  constructor(props) {
    super(props);
    this.state = {
      dollars: 10
    }
    this.onMouseLeaveHandler = this.onMouseLeaveHandler.bind(this);
    this.onTimeoutHandler = this.onTimeoutHandler.bind(this);
    this.onAjaxCallback = this.onAjaxCallback.bind(this);
    this.onClickHandler = this.onClickHandler.bind(this);
  }

  componentDidMount() {
    // Add custom event via `addEventListener`
    //
    // The list of supported React events does include `mouseleave`
    // via `onMouseLeave` prop
    //
    // However, we are not adding the event the `React way` - this will have
    // effects on how state mutates
    //
    // Check the list here - https://facebook.github.io/react/docs/events.html
    document.getElementById('testButton').addEventListener('mouseleave', this.onMouseLeaveHandler);

    // Add JS timeout
    //
    // Again,outside React `world` - this will also have effects on how state
    // mutates
    setTimeout(this.onTimeoutHandler, 10000);

    // Make AJAX request
    fetch('https://api.github.com/users')
      .then(this.onAjaxCallback);
  }

  onClickHandler = () => {
    console.log('State before (_onClickHandler): ' + JSON.stringify(this.state));
    this.setState({
      dollars: this.state.dollars + 10
    });
    console.log('State after (_onClickHandler): ' + JSON.stringify(this.state));
  }

  onMouseLeaveHandler = () => {
    console.log('State before (mouseleave): ' + JSON.stringify(this.state));
    this.setState({
      dollars: this.state.dollars + 20
    });
    console.log('State after (mouseleave): ' + JSON.stringify(this.state));
  }

  onTimeoutHandler = () => {
    console.log('State before (timeout): ' + JSON.stringify(this.state));
    this.setState({
      dollars: this.state.dollars + 30
    });
    console.log('State after (timeout): ' + JSON.stringify(this.state));
  }

  onAjaxCallback = (err, res) => {
    if (err) {
      console.log('Error in AJAX call: ' + JSON.stringify(err));
      return;
    }

    console.log('State before (AJAX call): ' + JSON.stringify(this.state));
    this.setState({
      dollars: this.state.dollars + 40
    });
    console.log('State after (AJAX call): ' + JSON.stringify(this.state));
  }

  render() {
    console.log('State in render: ' + JSON.stringify(this.state));

    return (
       <button
         id="testButton"
         onClick={this.onClickHandler}>
         'Click me'
      </button>
    );
  }
}

ReactDOM.render(
  <TestComponent />,
  document.getElementById('app')
);

解决 setState 函数异步的办法?

根据 React 官方文档,setState 函数实际上接收两个参数,其中第二个参数类型是一个函数,作为 setState 函数执行后的回调。通过传入回调函数的方式,React 可以保证传入的回调函数一定是在 setState 成功更新 this.state 之后再执行

_onClickHandler: function _onClickHandler() {
   console.log('State before (_onClickHandler): ' + JSON.stringify(this.state));
   this.setState({
   dollars: this.state.dollars + 10
   }, () => {
   console.log('Here state will always be updated to latest version!');
   console.log('State after (_onClickHandler): ' + JSON.stringify(this.state));
   });
}

更多关于 setState 的小知识

其实 setState 作为一个函数,本身是同步的。只是因为在 setState 的内部实现中,使用了 React updater 的 enqueueState 或者 enqueueCallback 方法,才造成了异步。

下面这段是 React 源码中 setState 的实现:

ReactComponent.prototype.setState = function(partialState, callback) {
  invariant(
    typeof partialState === 'object' ||
    typeof partialState === 'function' ||
    partialState == null,
    'setState(...): takes an object of state variables to update or a ' +
    'function which returns an object of state variables.'
  );
  this.updater.enqueueSetState(this, partialState);
  if (callback) {
    this.updater.enqueueCallback(this, callback, 'setState');
  }
};

而 updater 的这两个方法,又和 React 底层的 Virtual Dom (虚拟 DOM 树)的 diff 算法有紧密的关系,所以真正决定同步还是异步的其实是 Virtual DOM 的 diff 算法。

./setState.png

对着上面的流程图可以简单总结一下:

  • 将 setState 传入的 partialState 参数存储在当前组件实例的 state 暂存队列中
  • 判断当前 React 是否处于批量更新状态,如果是,将当前组件加入待更新的组件队列中
  • 如果未处于批量更新状态,将批量更新状态标识设置为 true,用事务再次调用前一步方法,保证当前组件加入到了待更新组件队列中
  • 调用事务的 waper 方法,遍历待更新组件队列依次执行更新。
  • 执行生命周期 componentWillReceiveProps
  • 将组件的 state 暂存队列中的 state 进行合并,获得最终要更新的 state 对象,并将队列置为空
  • 执行生命周期 componentShouldUpdate,根据返回值判断是否要继续更新
  • 执行生命周期 componentWillUpdate
  • 执行真正的更新,render
  • 执行生命周期 componentDidUpdate

Reference:

Passing a function to setState

我们已经提到过, setState 其实是异步的. 因为出于性能优化考虑, React 会将多次 setState 做一次批处理. 于是 setState 并不会在被调用之后立即改变我们的 state. 这就意味着你并不能依赖于在调用 setState 方法之后 state , 因为此时你并不能确认该 state 更新与否.

当然针对这个问题我们也有解决办法:用前一个 state(previous state) 作为需要传入函数的参数,将一个函数作为第二个参数传递给 setState ,这样做能保证你传入的函数需要取到的 state 一定会是被传入的 setState 执行之后的 state 。

问题:

// assuming this.state.count === 0
this.setState({count: this.state.count + 1});
this.setState({count: this.state.count + 1});
this.setState({count: this.state.count + 1});
// this.state.count === 1, not 3

解决办法:

this.setState((prevState, props) => ({
  count: prevState.count + props.increment
}));

and More:

// Passing object
this.setState({ expanded: !this.state.expanded });

// Passing function
this.setState(prevState => ({ expanded: !prevState.expanded }));

Reference:

Redux Reselect

在 React-Redux 的 connect(mapState)中使用 Reselect, 这能避免频繁的重新渲染的发生。

Reslect 会记录下上一次函数调用的结果并且当再次以相同方式调用时返回相同的结果(而不是创建一个一模一样的新结果). 只有当传入的参数不同时,才会产生新的结果。

Reference:

Dependency Injection

在 React 中,想做依赖注入(Dependency Injection)其实相当简单,具体有以下方法:

  • HOC——高阶组件(high-order component)
  • Context 传递(在最新的 React 官方文档中,Context 已经不太被官方推荐使用了:Why Not To Use Context

Reference:

React Lazy

React Lazy 是 16.6 之后添加的功能。需要配合 react.Suspense 使用。

类似社区实现的 react-loadable ,在之前我们是通过 webpack 的 require.ensure 或者自己封装 import() 来实现,本质上都是 code splitting

React hooks

Hooks 是 16.7 之后新添加的功能。

React Team 收了 recompose 演化成了 Hooks

简单理解Hooks 是一种函数,该函数允许你“勾住(hook into)”React 状态和来自函数组件的生命周期功能。

Hook 在类内部不起作用,它们允许你无需类就使用 React。

可以通过取巧的方式实现:hook 里面返回 class component,传入相应的 props

hooks 种类

React 内置了一些 hook,如 useState 当然也可以创建自定义的 Hooks 以在不同的组件当中复用状态。

根据 React 官方给出的文档,Hooks 主要分为以下几种:

  • State Hooks(可以实现 local state 或者 global state
  • Effect Hooks(可以实现 class 当中的一些 lifecycle
  • 自定义 Hooks(主要用来复用组件逻辑和状态

注意事项

官方定义了一些 rules of hooks,也就是使用 hooks 时候需要注意的内容:

  • 只能在顶层调用 Hook,不要在循环、条件或嵌套函数中调用 Hook。
  • 仅从 React 功能组件调用 Hook。不要从常规 JavaScript 函数调用 Hook。(还有另一个有效的地方来调用 Hook,即你的自定义 Hook。)

官方还提供了 linter plugin 来检查并自动执行这些规则

引入的背景

为什么要引入 Hooks ,大概有以下原因(摘自官方文档

  • 难以在组件之间重用有状态逻辑(组件树 🌲 过于臃肿
  • 复杂的组件变得难以理解(状态逻辑和抽象过多
  • 类让人和机器感到困惑(万恶的 this 和数据流的理解
  • 逐步采用策略(感觉是提供新的 idea,并不是抛弃 class component 的写法

截止至(2018 年 10 月 29 日 15:21:47),社区已经出现了很多关于 hooks 的库和 idea,印象较深的是这个 react-use ,看了一下源码对 hooks 的理解更多了一层

Reference:

React Fiber(浅析)

React 团队重写了核心算法 Reconciliation,优化在动画、手势等应用中的效果,通常将之前的 reconciler 称为 stack reconciler,重写后的称为 fiber reconciler,简称为 Fiber

关于 React Fiber Reconciliation 可以看Lin Clark - A Cartoon Intro to Fiber - React Conf 2017,或者看这篇文章《React Fiber》 用文字版解释了 Link Clark 的视频

Fiber 也称协程,这个概念在很多语言当中都存在,比如 go 的 Coroutine,还有前端开发者比较熟悉的 ES6 新增的 Generator,协程函数它只是一种控制流程的让出机制,可以中断和恢复函数

React Fiber 的思想和协程的概念是契合的: 🔴React 渲染的过程可以被中断,可以将控制权交回浏览器,让位给高优先级的任务,浏览器空闲后再恢复渲染

也可以理解 React 实现了Scheduler(调度器)来调度优先级不同的任务(任务包含 JS 的执行、页面的渲染、ui 事件的响应等等)

具体更详细的介绍可以查看我转载的这篇文章:转载:这可能是最通俗的-react-fiber 时间分片-打开方式

Reference: