[javascript] 배열에서 속성 값을 합산하는 더 좋은 방법

나는 이와 같은 것을 가지고있다 :

$scope.traveler = [
            {  description: 'Senior', Amount: 50},
            {  description: 'Senior', Amount: 50},
            {  description: 'Adult', Amount: 75},
            {  description: 'Child', Amount: 35},
            {  description: 'Infant', Amount: 25 },
];

이제이 배열의 총량을 얻으려면 다음과 같이하고 있습니다.

$scope.totalAmount = function(){
       var total = 0;
       for (var i = 0; i < $scope.traveler.length; i++) {
              total = total + $scope.traveler[i].Amount;
            }
       return total;
}

하나의 배열 만 있으면 쉽지만 다른 속성 이름을 가진 다른 배열이 있습니다.

이런 식으로 할 수 있다면 더 행복 할 것입니다.

$scope.traveler.Sum({ Amount });

그러나 나는 이것을 다음과 같이 재사용 할 수있는 방법으로 이것을 통과하는 방법을 모른다.

$scope.someArray.Sum({ someProperty });

대답

@ gruff-bunny 제안을 사용하기로 결정 했으므로 원시 객체 (배열)의 프로토 타이핑을 피하십시오

방금 배열의 유효성을 검사하는 그의 답변을 약간 수정했으며 합계 값이 null이 아닙니다. 이것은 최종 구현입니다.

$scope.sum = function (items, prop) {
    if (items == null) {
        return 0;
    }
    return items.reduce(function (a, b) {
        return b[prop] == null ? a : a + b[prop];
    }, 0);
};



답변

업데이트 된 답변

배열 프로토 타입에 함수를 추가하는 모든 단점으로 인해이 질문에 원래 요청 된 구문과 구문을 유지하는 대안을 제공하기 위해이 답변을 업데이트하고 있습니다.

class TravellerCollection extends Array {
    sum(key) {
        return this.reduce((a, b) => a + (b[key] || 0), 0);
    }
}
const traveler = new TravellerCollection(...[
    {  description: 'Senior', Amount: 50},
    {  description: 'Senior', Amount: 50},
    {  description: 'Adult', Amount: 75},
    {  description: 'Child', Amount: 35},
    {  description: 'Infant', Amount: 25 },
]);

console.log(traveler.sum('Amount')); //~> 235

원래 답변

배열이므로 배열 프로토 타입에 함수를 추가 할 수 있습니다.

traveler = [
    {  description: 'Senior', Amount: 50},
    {  description: 'Senior', Amount: 50},
    {  description: 'Adult', Amount: 75},
    {  description: 'Child', Amount: 35},
    {  description: 'Infant', Amount: 25 },
];

Array.prototype.sum = function (prop) {
    var total = 0
    for ( var i = 0, _len = this.length; i < _len; i++ ) {
        total += this[i][prop]
    }
    return total
}

console.log(traveler.sum("Amount"))

바이올린 : http://jsfiddle.net/9BAmj/


답변

나는이 질문에 대한 대답이 허용된다는 것을 알고 있지만 array.reduce 을 사용하는 대안을 사용 하여 배열을 합산하는 것이 축소의 전형적인 예라는 것을 알았습니다.

$scope.sum = function(items, prop){
    return items.reduce( function(a, b){
        return a + b[prop];
    }, 0);
};

$scope.travelerTotal = $scope.sum($scope.traveler, 'Amount');

Fiddle


답변

또 다른 특징은 nativeJavaScript 기능 MapReduce빌드를위한 것입니다 (Map 및 Reduce는 여러 언어로 제공되는 강국입니다).

var traveler = [{description: 'Senior', Amount: 50},
                {description: 'Senior', Amount: 50},
                {description: 'Adult', Amount: 75},
                {description: 'Child', Amount: 35},
                {description: 'Infant', Amount: 25}];

function amount(item){
  return item.Amount;
}

function sum(prev, next){
  return prev + next;
}

traveler.map(amount).reduce(sum);
// => 235;

// or use arrow functions
traveler.map(item => item.Amount).reduce((prev, next) => prev + next);

참고 : 더 작은 기능을 분리하여 다시 사용할 수 있습니다.

// Example of reuse.
// Get only Amounts greater than 0;

// Also, while using Javascript, stick with camelCase.
// If you do decide to go against the standards, 
// then maintain your decision with all keys as in...

// { description: 'Senior', Amount: 50 }

// would be

// { Description: 'Senior', Amount: 50 };

var travelers = [{description: 'Senior', amount: 50},
                {description: 'Senior', amount: 50},
                {description: 'Adult', amount: 75},
                {description: 'Child', amount: 35},
                {description: 'Infant', amount: 0 }];

// Directly above Travelers array I changed "Amount" to "amount" to match standards.

function amount(item){
  return item.amount;
}

travelers.filter(amount);
// => [{description: 'Senior', amount: 50},
//     {description: 'Senior', amount: 50},
//     {description: 'Adult', amount: 75},
//     {description: 'Child', amount: 35}];
//     Does not include "Infant" as 0 is falsey.


답변

나는 항상 프로토 타입 방법을 변경하고 라이브러리를 추가하는 것을 피하므로 이것이 내 솔루션입니다.

배열 프로토 타입 축소 방법을 사용하면 충분합니다

// + operator for casting to Number
items.reduce((a, b) => +a + +b.price, 0);


답변

나는 이것에 내 2 센트를 떨어 뜨릴 것이라고 생각했다 : 이것은 외부 변수에 의존하지 않고 항상 순수하게 기능 해야하는 작업 중 하나입니다. 몇 사람은 이미 좋은 대답을했으며 사용 reduce방법은 여기입니다.

우리 대부분은 이미 ES2015 구문을 사용할 여유가 있으므로 제 제안은 다음과 같습니다.

const sumValues = (obj) => Object.keys(obj).reduce((acc, value) => acc + obj[value], 0);

우리가있는 동안 우리는 그것을 불변 함수로 만들고 있습니다. 어떤 reduce값으로 시작 : 여기서하고있는 것은 단순히 이것이다 0누산기에 대한, 그리고 현재 루프 항목의 값을 추가합니다.

기능 프로그래밍과 ES2015를위한 예! 🙂


답변

향상된 가독성과 사용에 대한 대안 MapReduce:

const traveler = [
    {  description: 'Senior', amount: 50 },
    {  description: 'Senior', amount: 50 },
    {  description: 'Adult', amount: 75 },
    {  description: 'Child', amount: 35 },
    {  description: 'Infant', amount: 25 },
];

const sum = traveler
  .map(item => item.amount)
  .reduce((prev, curr) => prev + curr, 0);

재사용 가능한 기능 :

const calculateSum = (obj, field) => obj
  .map(items => items.attributes[field])
  .reduce((prev, curr) => prev + curr, 0);


답변

다음을 수행 할 수 있습니다.

$scope.traveler.map(o=>o.Amount).reduce((a,c)=>a+c);