示例代码:https://github.com/d6u/example-redux-update-nested-props/blob/master/one-connect/index.js

观看现场演示:http://d6u.github.io/example-redux-update-nested-props/one-connect.html

How to optimize small updates to props of nested component?

我有上述组件,回购和回购 list .我想更新第一次回购的标签(Line 14).所以我派出了UPDATE_TAG行动.在我实现shouldComponentUpdate之前,调度大约需要200毫秒,这是预期的,因为我们正在浪费大量时间来区分没有改变的<Repo/>秒.

添加shouldComponentUpdate后,调度大约需要30毫秒.在生产完成后,我们会做出react .js,更新只需17毫秒.这要好得多,但Chrome开发控制台中的时间线视图仍然显示jank frame(超过16.6ms).

enter image description here

想象一下,如果我们有很多这样的更新,或者<Repo/>比当前的更新更复杂,我们将无法维持60fps.

My question is, for such small updates to a nested component's props, is there a more efficient and canonical way to update the content? Can I still use Redux?

我找到了一个解决方案,用一个可观察的内部减速器替换每tags个.差不多

// inside reducer when handling UPDATE_TAG action
// repos[0].tags of state is already replaced with a Rx.BehaviorSubject
get('repos[0].tags', state).onNext([{
  id: 213,
  text: 'Node.js'
}]);

然后我用https://github.com/jayphelps/react-observable-subscribe在回购组件中订阅它们的值.这很有效.即使开发了React,每次调度也只需5毫秒.js.但我觉得这是Redux的反模式.

更新1

我遵循了丹·阿布拉莫夫的答案和normalized my stateupdated connect components中的建议

新的州形态是:

{
    repoIds: ['1', '2', '3', ...],
    reposById: {
        '1': {...},
        '2': {...}
    }
}

我在时间the initial rendering的基础上加了console.time.

但是,性能比以前差(包括初始渲染和更新).(来源:https://github.com/d6u/example-redux-update-nested-props/blob/master/repo-connect/index.js,现场演示:http://d6u.github.io/example-redux-update-nested-props/repo-connect.html)

// With dev build
INITIAL: 520.208ms
DISPATCH: 40.782ms

// With prod build
INITIAL: 138.872ms
DISPATCH: 23.054ms

enter image description here

我认为每<Repo/>个连接都有很多开销.

更新2

根据Dan的更新答案,我们必须返回connectmapStateToProps个参数,而不是返回一个函数.你可以看看丹的答案.我还更新了the demos条.

下面,我的电脑的性能要好得多.为了好玩,我还添加了我所说的方法(sourcedemo)(seriously don't use it, it's for experiment only)中的副作用.

// in prod build (not average, very small sample)

// one connect at root
INITIAL: 83.789ms
DISPATCH: 17.332ms

// connect at every <Repo/>
INITIAL: 126.557ms
DISPATCH: 22.573ms

// connect at every <Repo/> with memorization
INITIAL: 125.115ms
DISPATCH: 9.784ms

// observables + side effect in reducers (don't use!)
INITIAL: 163.923ms
DISPATCH: 4.383ms

更新3

刚刚根据"每一次都与记忆联系"增加了react-virtualized example

INITIAL: 31.878ms
DISPATCH: 4.549ms

推荐答案

我不确定const App = connect((state) => state)(RepoList)是从哪里来的

别这样!它会扼杀任何性能优化,因为TodoApp会在每次操作后重新启动.

我们不建议使用这种模式.相反,每一个都专门连接<Repo>,以便在其mapStateToProps中读取自己的数据."tree-view"的例子展示了如何做到这一点.

如果将状态形状设置为normalized以上(现在都是嵌套的),则可以将repoIdsreposById分开,然后仅在repoIds发生变化时才重新渲染RepoList.这样,对单个回购协议的更改不会影响列表本身,只有相应的Repo将被重新呈现.This pull request可能会让你知道这是怎么回事."real-world"示例展示了如何编写处理规范化数据的减缩器.

请注意,为了真正从规范化树提供的性能中获益,您需要完全像this pull request一样,将mapStateToProps()工厂传递给connect():

const makeMapStateToProps = (initialState, initialOwnProps) => {
  const { id } = initialOwnProps
  const mapStateToProps = (state) => {
    const { todos } = state
    const todo = todos.byId[id]
    return {
      todo
    }
  }
  return mapStateToProps
}

export default connect(
  makeMapStateToProps
)(TodoItem)

这一点之所以重要,是因为我们知道ID永远不会改变.使用ownProps会带来性能损失:当外部props 发生变化时,内部props 必须重新计算.然而,使用initialOwnProps不会招致这种惩罚,因为它只使用了一次.

您的示例的快速版本如下所示:

import React from 'react';
import ReactDOM from 'react-dom';
import {createStore} from 'redux';
import {Provider, connect} from 'react-redux';
import set from 'lodash/fp/set';
import pipe from 'lodash/fp/pipe';
import groupBy from 'lodash/fp/groupBy';
import mapValues from 'lodash/fp/mapValues';

const UPDATE_TAG = 'UPDATE_TAG';

const reposById = pipe(
  groupBy('id'),
  mapValues(repos => repos[0])
)(require('json!../repos.json'));

const repoIds = Object.keys(reposById);

const store = createStore((state = {repoIds, reposById}, action) => {
  switch (action.type) {
  case UPDATE_TAG:
    return set('reposById.1.tags[0]', {id: 213, text: 'Node.js'}, state);
  default:
    return state;
  }
});

const Repo  = ({repo}) => {
  const [authorName, repoName] = repo.full_name.split('/');
  return (
    <li className="repo-item">
      <div className="repo-full-name">
        <span className="repo-name">{repoName}</span>
        <span className="repo-author-name"> / {authorName}</span>
      </div>
      <ol className="repo-tags">
        {repo.tags.map((tag) => <li className="repo-tag-item" key={tag.id}>{tag.text}</li>)}
      </ol>
      <div className="repo-desc">{repo.description}</div>
    </li>
  );
}

const ConnectedRepo = connect(
  (initialState, initialOwnProps) => (state) => ({
    repo: state.reposById[initialOwnProps.repoId]
  })
)(Repo);

const RepoList = ({repoIds}) => {
  return <ol className="repos">{repoIds.map((id) => <ConnectedRepo repoId={id} key={id}/>)}</ol>;
};

const App = connect(
  (state) => ({repoIds: state.repoIds})
)(RepoList);

console.time('INITIAL');
ReactDOM.render(
  <Provider store={store}>
    <App/>
  </Provider>,
  document.getElementById('app')
);
console.timeEnd('INITIAL');

setTimeout(() => {
  console.time('DISPATCH');
  store.dispatch({
    type: UPDATE_TAG
  });
  console.timeEnd('DISPATCH');
}, 1000);

请注意,我在ConnectedRepo中将connect()改为使用initialOwnProps而不是ownProps的工厂.这让React Redux跳过所有props 重新判断.

我还删除了<Repo>上不必要的shouldComponentUpdate(),因为React Redux负责在connect()中实现它.

在我的测试中,这种方法优于之前的两种方法:

one-connect.js: 43.272ms
repo-connect.js before changes: 61.781ms
repo-connect.js after changes: 19.954ms

最后,如果你需要显示如此大量的数据,它无论如何都无法显示在屏幕上.实际上,在不显示上千行的情况下,显示上千行的效果更好.


我找到了一个解决方案,用一个可观察的内部减速器替换每个标签.

如果它有副作用,它不是Redux减量剂.这可能有效,但我建议将这样的代码放在Redux之外,以避免混淆.Redux reducer必须是纯函数,它们不能调用onNext个主题.

Reactjs相关问答推荐

使用REACT-RUTER-V6和ZUSTANDreact 中的身份验证

App.js中的H2组件不会动态呈现.这可能是什么问题?

获取未捕获的错误:Gatsby生产版本上的最小化react 错误#418

在url中使用MongoDB对象ID被认为是不好的做法?

shadcn输入+窗体+Zod;一个部件正在改变要被控制的不受控制的输入;

无法在 NextJS 中获取嵌套对象

React Wordpress Apache 在索引之外重新加载 -> 未找到页面

显示我是否加入聊天应用程序时出现问题

当 React 中的状态发生变化时如何停止重新加载谷歌 map ?

需要先填写依赖输入字段,以便其他人工作

为什么 setState 在 React 中不是异步的?

表单错误后 Ionic React 无法 Select 单选按钮

如何禁用来自 React Native API 的某些特定项目的可touch 不透明度

Cypress - 替换 React 项目中的变量

Mui CardMedia 的图片组件

将 Material UI 菜单渲染为

如何在 Cypress E2E 的站点上测试react 组件?

如何在 React 中的 Material Ui 多选中设置默认值?

react 路由路由加载器不适用于嵌套组件

gtag:为什么谷歌分析即使没有被授权也会收集数据