女王控的博客

优化 React 性能

翻译转载:Optimize React Performance

如何提高生产中的 React 性能,避免常见的错误

您是否知道 key 在列表中使用不正确可以将 DOM 操作增加 10 倍以上?据了解,React 的 Virtual DOM 实现速度非常快。但是,即使使用 React 的内置性能,也有一些 UI 可能开始感觉迟缓的情况。造成业绩不佳的罪魁祸首是产生了太多渲染和状态同步。我们将介绍 4 种技术,以帮助确保您的应用程序得到优化,并提供快速的用户体验。

  1. key 正确使用
  2. 管理 shouldComponentUpdate
  3. 扩展 PureComponent 而不是 Component
  4. 建立生产

衡量效率

在深入研究代码之前,了解您正在优化的内容以及它所带来的影响是非常重要的。我建议使用的两个工具是 Lighthouse 和 react-addons-perf。这两个工具都可以通过 NPM(安装12),并都提供一个浏览器扩展(12)。我个人建议使用 npm for Lighthouse,因为结果似乎更可靠,并使用 React Perf 的 Chrome 扩展。以下是来自 Lighthouse 的输出示例,我们将在以下部分查看 React Perf 的结果。

1 P0wwSgmIoA48qFnsMbSamg

注意:Lighthouse 从版本 60 开始可以通过 Chrome 本地浏览,而 react-addons-perf 将不再与 React 16.0 +兼容,但以下结果对任何版本的 React 仍然有效。

正确的使用 key

key:React 中在应用程序生命周期的任何时刻唯一标识列表中的元素。如果 DOM 被更新,React 能够识别哪些项目已经改变。因此,key 组件上的值必须是唯一的,并且与列表中的同级组件相关。这是 React 开发人员倾向于使用的一个属性,但是不论经验多少的 React 开发人员在生成列表时可能不会真正理解使用该属性。两个常见的错误是使用 index 期间 map(element, index) 或使用 Math.random() 生成一个 key。最好的解决方案是与列表项关联 UUID,但任何唯一标识符都可以。在列表中您将使用 key 的典型场景如下。

js 复制代码
function ListItem(props) {
  // Correct! There is no need to specify the key here:
  return <li>{props.value}</li>;
}

function UserList(props) {
  const users = props.users;
  const listItems = users.map((user) => (
    // Correct! Key should be specified inside the array.
    <ListItem key={user.id.toString()} value={user.name} />
  ));
  return <ul>{listItems}</ul>;
}

const users = [{ id: 1, name: 'Khaleesi' }, { id: 2, name: 'Jon' }, { id: 3, name: 'Arya' }];

<UserList users={users} />;

由于用户 ID 是唯一的,我们可以确信地使用它作为列表项的关键字。对于更实际的应用程序,我创建了一个 Postmate 的 feed 的模拟版本,当点击时从 DOM 中删除了一个商店。

1 hJJK3sBRN2g3NbXDPHTF9A

1 K9z91UAqGDb8pbcU1GRcPg

使用 react-addons-perf Chrome 扩展,我计算了使用 UUID 作为关键字以及使用 Math.random() 一个关键字来移除一个元素的操作次数。结果表明,使用唯一一致的密钥只需要 28 个操作来移除项目,而随机生成的密钥需要超过25 倍的数目操作才能执行完全相同的任务。谨慎用你的key

管理 shouldComponentUpdate

管理组件生命周期允许您准确指定组件应该呈现的时间。既然您作为开发人员了解您的应用程序并不知道您的应用程序,您就可以确定组件是否需要重新渲染。渲染和重绘是 React 中最昂贵的操作,因此根据您的领域知识消除任何不必要的调用可以为您的性能和用户体验带来巨大的改进。

使用 React.PureComponent 而不是 React.Component

React 提供 PureComponent 了管理 shouldComponentUpdate 开发人员的生命周期钩子。PureComponent 将做所有道具/状态的浅层比较,并且只会在任何道具/状态发生变化时进行渲染。大多数情况下,您可以放弃使用 PureComponent 而不是自己处理。它确实带有一些仔细的考虑,可能导致讨厌的错误。在将对象分配给变量时,JavaScript 使用指针。这意味着变量指向内存中的特定位置,而不是指向对象本身的任何特定位置。如果更新对象的属性,指针将保持不变,因此浅比较将无法识别更新。为了避免这种情况,您可以

  1. 直接将变量传递给组件,而不是传递给对象
  2. 包含不可变的数据结构。

以下示例使用 PureComponent 将获得与我们的示例相同的结果 shouldComponentUpdate。

js 复制代码
class CounterButton extends React.PureComponent {
  constructor(props) {
    super(props);
    this.state = { count: 1 };
    this.handleClick = this.handleClick.bind(this);
  }

  handleClick() {
    this.setState((state) => ({ count: state.count + 1 }));
  }

  render() {
    return (
      <button color={this.props.color} onClick={this.handleClick}>
        Count: {this.state.count}
      </button>
    );
  }
}

最后一个考虑因素是功能组件还没有针对 React 进行优化,但是团队已经答应在将来这样做。如果您在应用程序中确定了需要优化的区域,则最好使用类组件来确保获得所需的性能。

建立生产

建设生产看起来很明显,但重要的是你检查你的过程,以确保它编译使用 NODE_ENV = ‘production’。React 包含许多警告,可帮助您调试在生产中被剥离的内容。这些警告使开发更友好,但也使得 React 的构建变得越来越慢。生产 React 构建比开发构建渲染速度快 2-8 倍。

以下 Webpack 插件将提供所需的生产版本:

js 复制代码
plugins: [
  new webpack.DefinePlugin({
    'process.env': {
      NODE_ENV: JSON.stringify('production')
    }
  }),
  new webpack.optimize.UglifyJsPlugin()
  'transform-react-constant-elements',
  'transform-react-inline-elements',
  ...otherPlugins
]

在 DefinePlugin 确保 NODE_ENV 设置是否正确,以及 UglifyJsPlugin 压缩的 JavaScript。所述 transform-react-constant-elements 移动常数元件以更高的范围,和 transform-react-inline-elements 替换天然 React.createElement() 具有更优化的版本。请注意,这些只能在生产中使用,因为它们会产生模糊的错误,使开发调试变得困难。您可以使用 React Developer for Chrome 来评估您是否使用 React 的生产版本。

评论

阅读上一篇

Webpack3代码拆分
2017-12-13 14:50:16

阅读下一篇

React基础知识第三部分
2017-12-11 23:45:34
0%