オブジェクトのJavascript reduce()


182

Array reduce()から1つの値を取得するための素敵なArrayメソッドがあります。例:

[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;
});

ただし、Objectにはreduce()メソッドが実装されていないようです。


1
使っていUnderscore.jsますか?
Sethen 2013

いいえ。Underscoreはオブジェクトの削減を提供しますか?
Pavel S.

思い出せない。私はそれがreduce方法を持っていることを知っています。そこでチェックします。しかし、解決策はそれほど難しくはありません。
Sethen 2013

1
@Sethen Maleno、@ Pavel:はい_、オブジェクトの削減があります。それが偶然に機能したのか、オブジェクトのサポートが意図的なものであったのかはfor..inわかりませんが、実際には、この質問の例のようにオブジェクトを渡すことができます。
Roatin Marth 2013

回答:


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()です-今すぐ編集
Jonathan Wood

281

1つのオプションは次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);

これにより、初期値を指定する必要があります。そうしないと、最初のラウンドはになります'a' + 2

結果をオブジェクト({ value: ... })として取得する場合は、毎回オブジェクトを初期化して返す必要があります。

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

15
いい答えですが、ここではキーではなく値に関心があるため、Object.keysの代わりにObject.valuesを使用する方が読みやすくなっています。次のようになります。Object.values(o).reduce((total、current)=> total + current.value、0);
ミナルーク

3
Object.valuesのブラウザーサポートはObject.keysよりはるかに劣っていますが、ポリフィルを使用する場合、または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 2017

1
const [key、value] =ペア; これを見たことがない!
Martin Meeser、

9
@ martin-meeser-これは構造化解除と呼ばれます。次のように変更function (total, pair)して、この行を省略することもできますfunction (total, [key, value])
JakubZawiślakMar

一方でentries、このオーバーを行うためのクリーンな方法でkeys、それはパフォーマンスが低いです。hackernoon.com/...
robdonn

@faboulaws Object.entries(o); // [["a"、{value:1}]、["b"、{value:2}]、["c"、{value:3}]]を返す
Thomas

19

まず第一に、reduceの以前の値が何であるかを十分に理解していません。

あなたが持っている疑似コードではreturn previous.value + current.value、したがって、previous値はオブジェクトではなく次の呼び出しの数値になります。

2 reduceつ目は、オブジェクトのメソッドではなく配列メソッドであり、オブジェクトのプロパティを反復するときに順序に依存することはできません(https://developer.mozilla.org/en-US/docs/を参照)。JavaScript / Reference / Statements / for ... inでは、これは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;
});

ファットアロー関数の構文を備えたES6のPS(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'Joy it、guys !! ;-)


-1、将来のすべてのオブジェクトに新しい列挙可能なプロパティがあります:jsfiddle.net/ygonjooh
Johan


1
このように基本プロトタイプを変更しないでください。これは、同じコードベースで作業する将来の開発者にとって多くの問題につながる可能性があります
adam.k

はい、それはこのソリューションは、6年前に書かれていない、あまりにも関連する今、ちょうど心に留めておく。例えば、それはより良い利用になりました「パッチ適用猿」であるObject.entries()2021年
user1247458

2

ジェネレータ式(現在すべてのブラウザで長年サポートされており、Nodeでもサポートされています)を使用して、キーと値のペアをリストで取得できます。

>>> 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()で配列に変換できますとして縮小できます。ただし、中間配列を作成せずにオブジェクトを削減することもできます。

オブジェクトを操作するための小さなヘルパーライブラリodictを作成しました。

npm install --save odict

Array.prototype.reduce()と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

回答ではまだ確認されていないため、Underscore 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

役に立たない
Chris Dolphin

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)
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.