JavaScript JavaScript () на Object


182

Є хороший метод масиву, reduce()щоб отримати одне значення з масиву. Приклад:

[0,1,2,3,4].reduce(function(previousValue, currentValue, index, array){
  return previousValue + currentValue;
});

Який найкращий спосіб досягти того ж з предметами? Я хотів би зробити це:

{ 
    a: {value:1}, 
    b: {value:2}, 
    c: {value:3} 
}.reduce(function(previous, current, index, array){
  return previous.value + current.value;
});

Однак, здається, в «Об’єкті» не було reduce()реалізовано жодного методу.


1
Ви використовуєте Underscore.js?
Сетен

Ні. Чи передбачено підкреслення для зменшення об’єктів?
Павло С.

Я не можу згадати. Я знаю, що у нього є reduceметод. Я би там перевірив. Хоча рішення не здається таким складним.
Сетен

1
@Sethen Maleno, @Pavel: так _, зменшення об’єктів є. Не впевнений, чи працює він випадково або якщо підтримка об’єктів була навмисною, але дійсно ви можете передати об’єкт, як у прикладі цього питання, і це буде (концептуально) for..in, викликаючи вашу функцію ітератора зі значеннями, знайденими на кожному ключі.
Roatin Marth

Відповіді:


55

Те, що ви насправді хочете в цьому випадку, - це Object.values. Ось стисла реалізація ES6 з урахуванням цього:

const add = {
  a: {value:1},
  b: {value:2},
  c: {value:3}
}

const total = Object.values(add).reduce((t, {value}) => t + value, 0)

console.log(total) // 6

або просто:

const add = {
  a: 1,
  b: 2,
  c: 3
}

const total = Object.values(add).reduce((t, n) => t + n)

console.log(total) // 6

Це Array.prototype.values ​​(), яку ви пов’язали - редагував зараз
Джонатан Вуд

281

Одним з варіантів було б :reducekeys()

var o = { 
    a: {value:1}, 
    b: {value:2}, 
    c: {value:3} 
};

Object.keys(o).reduce(function (previous, key) {
    return previous + o[key].value;
}, 0);

За допомогою цього ви хочете вказати початкове значення, або 1-й раунд буде 'a' + 2.

Якщо ви хочете, щоб результат був об'єктом ( { value: ... }), вам доведеться ініціалізувати та повертати об'єкт кожен раз:

Object.keys(o).reduce(function (previous, key) {
    previous.value += o[key].value;
    return previous;
}, { value: 0 });

15
Приємна відповідь, але легше читати використовувати Object.values ​​замість Object.keys, тому що нас турбують значення, а не клавіші. Він повинен бути таким: Object.values ​​(o) .reduce ((загальний, поточний) => total + current.value, 0);
Міна Люк

3
Object.values ​​має набагато гіршу підтримку браузера, ніж Object.keys, але це може бути проблемою, якщо ви використовуєте polyfill або транслітуєте з Babel
duhaime

саме я використовував це в клавішах, які отримали буряк із моделі mongo, і я передав порожній об’єкт як початкове значення в результаті зменшення клавіш, і стільки, скільки я використовував @babel/plugin-proposal-object-rest-spreadплагін, я поширював значення акумулятора в повернення зменшення, і це працює як шарм, але мені було цікаво, якщо я роблю щось не так, тому що я передавав предмет до початкового значення зменшення, і ваша відповідь довела мені, що я зробив правильно!
a_m_dev

55

Впровадження ES6: Object.entries ()

const o = {
  a: {value: 1},
  b: {value: 2},
  c: {value: 3}
};

const total = Object.entries(o).reduce(function (total, pair) {
  const [key, value] = pair;
  return total + value;
}, 0);

3
Object.entries(o); // returns [['value',1],['value',2],['value',3]]
faboulaws

1
const [ключ, значення] = пара; Я ніколи цього не бачив!
Мартін Мізер

9
@ martin-meeser - це називається руйнуванням. Ми можемо навіть опустити цю лінію, змінивши function (total, pair)наfunction (total, [key, value])
Якуб Завіслак

Хоча entriesце більш чистий спосіб зробити це keys, але він менш ефективний. hackernoon.com/…
robdonn

@faboulaws Object.entries (o); // повертає [["a", {значення: 1}], ["b", {значення: 2}], ["c", {значення: 3}]]
Томас

19

Перш за все, ви не зовсім розумієте, що таке попереднє значення зменшення .

У вас є псевдокод return previous.value + current.value, тому previousзначенням буде наступне дзвінок, а не об'єкт.

По-друге, reduceце метод масиву, а не об’єкт, і ви не можете розраховувати на порядок, коли ви повторюєте властивості об'єкта (див. Https://developer.mozilla.org/en-US/docs/ JavaScript / Довідка / Заяви / для ... в , це також застосовується до Object.keys ); тому я не впевнений, чи reduceмає сенс застосування над об’єктом.

Однак якщо замовлення не важливе, ви можете мати:

Object.keys(obj).reduce(function(sum, key) {
    return sum + obj[key].value;
}, 0);

Або ви можете просто зіставити значення об'єкта:

Object.keys(obj).map(function(key) { return this[key].value }, obj).reduce(function (previous, current) {
    return previous + current;
});

PS в ES6 з синтаксисом функції жирової стрілки (вже в Firefox Nightly), ви можете трохи зменшити:

Object.keys(obj).map(key => obj[key].value).reduce((previous, current) => previous + current);

3

1:

[{value:5}, {value:10}].reduce((previousValue, currentValue) => { return {value: previousValue.value + currentValue.value}})

>> Object {value: 15}

2:

[{value:5}, {value:10}].map(item => item.value).reduce((previousValue, currentValue) => {return previousValue + currentValue })

>> 15

3:

[{value:5}, {value:10}].reduce(function (previousValue, currentValue) {
      return {value: previousValue.value + currentValue.value};
})

>> Object {value: 15}

2

Розширити Object.prototype.

Object.prototype.reduce = function( reduceCallback, initialValue ) {
    var obj = this, keys = Object.keys( obj );

    return keys.reduce( function( prevVal, item, idx, arr ) {
        return reduceCallback( prevVal, item, obj[item], obj );
    }, initialValue );
};

Зразок використання.

var dataset = {
    key1 : 'value1',
    key2 : 'value2',
    key3 : 'value3'
};

function reduceFn( prevVal, key, val, obj ) {
    return prevVal + key + ' : ' + val + '; ';
}

console.log( dataset.reduce( reduceFn, 'initialValue' ) );
'Output' == 'initialValue; key1 : value1; key2 : value2; key3 : value3; '.

n'Doy it, хлопці !! ;-)


-1, тепер у вас є нове чисельне майно для всіх майбутніх об’єктів: jsfiddle.net/ygonjooh
Йохан


1
Будь ласка, не змінюйте подібний базовий прототип. Це може спричинити багато проблем для майбутніх розробників, які працюють в одній і тій же кодовій базі
adam.k

Так, це "патч" мавпи "Це рішення було написане 6 років тому і не надто актуальне зараз, просто майте на увазі. Наприклад, краще буде користуватися Object.entries()в 2021 році
користувач1247458

2

Ви можете використовувати генераторний вираз (підтримується у всіх браузерах роками та у вузлі), щоб отримати пари ключових значень у списку, на який можна зменшити:

>>> a = {"b": 3}
Object { b=3}

>>> [[i, a[i]] for (i in a) if (a.hasOwnProperty(i))]
[["b", 3]]

2

Об'єкт можна перетворити на масив за допомогою: Object.entries () , Object.keys () , Object.values ​​() , а потім зменшити як масив. Але ви також можете зменшити об'єкт, не створюючи проміжний масив.

Я створив невеличку бібліотеку- одікт для роботи з об'єктами.

npm install --save odict

Він має reduceфункцію, яка дуже працює як Array.prototype.reduce () :

export const reduce = (dict, reducer, accumulator) => {
  for (const key in dict)
    accumulator = reducer(accumulator, dict[key], key, dict);
  return accumulator;
};

Ви також можете призначити його:

Object.reduce = reduce;

так як цей метод дуже корисний!

Тож відповідь на ваше запитання буде:

const result = Object.reduce(
  {
    a: {value:1},
    b: {value:2},
    c: {value:3},
  },
  (accumulator, current) => (accumulator.value += current.value, accumulator), // reducer function must return accumulator
  {value: 0} // initial accumulator value
);

1

Якщо ви можете використовувати масив, використовуйте масив, довжина та порядок масиву - це половина його вартості.

function reducer(obj, fun, temp){
    if(typeof fun=== 'function'){
        if(temp== undefined) temp= '';
        for(var p in obj){
            if(obj.hasOwnProperty(p)){
                temp= fun(obj[p], temp, p, obj);
            }
        }
    }
    return temp;
}
var O={a:{value:1},b:{value:2},c:{value:3}}

reducer(O, function(a, b){return a.value+b;},0);

/ * повернене значення: (Кількість) 6 * /


1

Зробити це не дуже складно:

function reduceObj(obj, callback, initial) {
    "use strict";
    var key, lastvalue, firstIteration = true;
    if (typeof callback !== 'function') {
        throw new TypeError(callback + 'is not a function');
    }   
    if (arguments.length > 2) {
        // initial value set
        firstIteration = false;
        lastvalue = initial;
    }
    for (key in obj) {
        if (!obj.hasOwnProperty(key)) continue;
        if (firstIteration)
            firstIteration = false;
            lastvalue = obj[key];
            continue;
        }
        lastvalue = callback(lastvalue, obj[key], key, obj);
    }
    if (firstIteration) {
        throw new TypeError('Reduce of empty object with no initial value');
    }
    return lastvalue;
}

Дія:

var o = {a: {value:1}, b: {value:2}, c: {value:3}};
reduceObj(o, function(prev, curr) { prev.value += cur.value; return prev;}, {value:0});
reduceObj(o, function(prev, curr) { return {value: prev.value + curr.value};});
// both == { value: 6 };

reduceObj(o, function(prev, curr) { return prev + curr.value; }, 0);
// == 6

Ви також можете додати його до прототипу Object:

if (typeof Object.prototype.reduce !== 'function') {
    Object.prototype.reduce = function(callback, initial) {
        "use strict";
        var args = Array.prototype.slice(arguments);
        args.unshift(this);
        return reduceObj.apply(null, args);
    }
}

0

Оскільки це ще не було підтверджено у відповіді, підкреслення reduceтакож працює на це.

_.reduce({ 
    a: {value:1}, 
    b: {value:2}, 
    c: {value:3} 
}, function(prev, current){
    //prev is either first object or total value
    var total = prev.value || prev

    return total + current.value
})

Зауважте, _.reduceповерне єдине значення (об'єкт або іншим чином), якщо в об'єкті списку є лише один елемент, без виклику функції ітератора.

_.reduce({ 
    a: {value:1} 
}, function(prev, current){
    //not called
})

//returns {value: 1} instead of 1

"Спробуйте цю іншу бібліотеку" не корисно
Grunion Shaftoe

Не корисний для вас
Кріс Дельфін

0

Спробуйте цю функцію зі стрілкою на вкладиші

Object.values(o).map(a => a.value, o).reduce((ac, key, index, arr) => ac+=key)

0

Спробуйте це. Він буде сортувати числа за іншими змінними.

const obj = {
   a: 1,
   b: 2,
   c: 3
};
const result = Object.keys(obj)
.reduce((acc, rec) => typeof obj[rec] === "number" ? acc.concat([obj[rec]]) : acc, [])
.reduce((acc, rec) => acc + rec)
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.