27

我正在寻找一个纯函数来修改我的不可变状态对象。作为参数给出的原始状态必须保持不变。这在使用像Redux这样的框架时特别有用,并且使在 javascript 中使用不可变对象变得更加容易。特别是因为使用Babel使用对象扩展运算符已经成为可能。

我没有找到比首先复制对象更好的方法,并且比分配/删除我想要的属性是这样的:

function updateState(state, item) {
  newState = {...state};
  newState[item.id] = item;
  return newState;
}

function deleteProperty(state, id) {
    var newState = {...state};
    delete newState[id];
    return newState;
}

感觉可以短一点

4

6 回答 6

71

对状态的操作,其中状态被认为是不可变的。

添加或更新属性的值

// ES6:
function updateState(state, item) {
    return Object.assign({}, state, {[item.id]: item});
}

// With Object Spread:
function updateState(state, item) {
  return {
     ...state,
     [item.id]: item
  };
}

删除属性

// ES6:
function deleteProperty(state, id) {
    var newState = Object.assign({}, state);
    delete newState[id];
    return newState; 
}

// With Object Spread:
function deleteProperty(state, id) {
    let  {[id]: deleted, ...newState} = state;
    return newState;
}

// Or even shorter as helper function:
function deleteProperty({[id]: deleted, ...newState}, id) {
    return newState;
}

// Or inline:
function deleteProperty(state, id) {
    return (({[id]: deleted, ...newState}) => newState)(state);
}
于 2016-04-11T15:36:53.410 回答
8

具有更多支持的 ES6 解决方案是Object.assign

const updateState = (state, item) => Object.assign({}, state, { [item.id]: item });
于 2016-04-11T15:40:57.667 回答
2

在地图函数中

要在映射函数中执行此过程(删除一个属性并在每个对象上添加一个新属性),给定一个对象数组 -

const myArrayOfObjects = [
    {id: 1, keyToDelete: 'nonsense'},
    {id: 2, keyToDelete: 'rubbish'}
];

删除该属性keyToDelete,并添加一个newKey具有该值的新键"someVar"

myArrayOfObjects.map(({ keyToDelete, ...item}) => { ...item, newKey:'someVar'});

将数组更新为

[
    {id: 1, newKey:'someVar'},
    {id: 2, newKey:'someVar'}
]

有关删除方法的更多信息,请参阅这篇很棒的帖子

于 2018-07-25T11:01:57.163 回答
1

(({[id]: deleted, ...state}) => state)(state)您可以使用一些库来做同样的事情,而不是编写难以阅读的样板代码(如上所述:) :

例如:

import {remove} from 'immutable-modify'

function updateState(state, item) {
   return remove(state, item.id)
}

它还支持任何嵌套更新:

import {set} from 'immutable-modify'

function updateState(state, item) {
   return set(state, 'user.products', (products) => ({
      ...products,
      items: products.items.concat(item),
      lastUpdate: Date.now()
   }))
}
于 2018-07-19T12:20:45.910 回答
1

尝试:

const { id, ...noId } = state;

并测试:

console.log(noId);
于 2019-03-05T05:30:40.933 回答
-2

从数组中删除项目,只需使用过滤器;)

CASE 'REMOVE_ITEM_SUCCESS':

      let items = state.items.filter(element => element._id !== action.id);

      return {
            ...state, 
            items
      }
于 2018-02-01T23:18:20.927 回答