如何在 Immutablejs 和 Redux 以及 Flux 和 React 中设置 Ember 之类的计算属性

IT技术 reactjs flux immutable.js redux
2021-05-24 13:08:51

我习惯于在Ember Object Model 中计算属性这是一种指定依赖于其他属性的计算属性的便捷方法。

fullName取决于firstNamelastName,我可以将计算属性设置为函数computeProperties并在computeProperties每次进行更改时调用

例子:

function computeFullName(state) {
  const fullName = state.get('firstName') + state.get('lastName');
  const nextState = state.set('fullName', fullName);
  return nextState;
}

function computeProperties(state) {
  const nextState = computeFullName(state);
  return nextState;
}

// store action handler
[handleActionX](state) {

  let nextState = state.set('firstName', 'John');
  nextState = state.set('lastName', 'Doe');

  nextState = computeProperties(nextState);

  return nextState;
}

有没有办法自动设置计算属性,这样我就不必每次都调用额外的函数。在 Redux 或 ImmutableJS 中。

4个回答

Redux 作者在这里!

按照 WildService 的建议使用 reselect 是可行的方法。我认为我们不会将它包含在核心中,因为 reselect 可以很好地完成它的工作,并且我们可以将它作为一个单独的库。

我想注意几点:

  • 即使使用重新选择,您也不希望在减速器中计算数据。选择器应该减速器管理的状态进行操作换句话说,选择器是 Redux 存储状态和组件之间的步骤——它们不在你的减速器中。保持 Redux 状态正常化是很重要的,这样它就很容易更新。

  • 我们实际上鼓励您在相关的reducer旁边定义选择器,这样当您更改状态形状时,您不必更改您的组件——它们将使用选择器代替。您可以在Flux 比较Redux 文件夹中看到这样的示例

  • 我们有一个文档页面介绍 reselect 并描述如何使用它来计算派生数据看看这个。

签出重新选择用于有效计算来自商店的派生数据的可组合纯函数。Afaik 计划在某个阶段将 reselect 的选择器滚动到 Redux 核心,如果它们被证明很受欢迎。在自述文件的底部也有一个使用 ImmutableJS 的示例。

要创建计算属性,您可以使用独立的 observable 库mobservable

var user = mobservable.props({
  firstName: 'John',
  lastName: 'Doe',
  fullName: function() {
    return this.firstName + this.lastName
  }
});

var nameViewer = mobservable.ObservingComponent(React.createClass({
   render: function() {
       return (<span>{user.fullName}</span>)
   }
});

这应该是它的要点,现在对 user.firstName 或 lastName 的任何更改都将重新呈现您的 nameViewer 组件。您可以进一步将其与 redux 之类的通量实现结合起来以更改数据,并通过您的组件树推送用户本身。但请注意,用户对象本身并不是一成不变的(在那种情况下,它毕竟不会被观察到 ;-))另请参阅这个琐碎稍微有趣的一些示例的小提琴。

像这样的事情怎么办?

export const getWidgetsWithComputedProps = widgets => {
  return widgets.map(w => getWidgetWithComputedProps(w));
};

export const selectWidgetType = widget => {
  switch (widget.type) {
    case 'line':
      return 'time-series';
    case 'pie':
    case 'bar':
      return 'cross-sectional';
    default:
      console.warn('Back up: that type of widget does not exist!', widget.type);
      return null;
  }
};

export const getWidgetWithComputedProps = createSelector(
  widget => widget,
  selectWidgetType,
  (widget, _type) => {
    return {...widget, _type}
  }
);