Javascript:使用特定键对JSON对象进行分组

时间:2015-10-12 09:27:56

标签: javascript json

我有以下JSON对象,并想通过OrderID合并它们,使这些项成为对象数组:

[
  {
    "OrderID":"999123",
    "ItemCode":"TED-072",
    "ItemQuantity":"1",
    "ItemPrice":"74.95",
  },
  {
    "OrderID":"999123",
    "ItemCode":"DY-FBBO",
    "ItemQuantity":"2",
    "ItemName":"DOIY Foosball Bottle Opener > Red",
    "ItemPrice":"34.95",
  }
]

我想知道如何在Javascript中合并同一订单上的商品......就像这样:

[{
    "OrderID": "999123",
        "Items": [{
        "ItemCode": "DY-FBBO",
            "ItemQuantity": "2",
            "ItemName": "DOIY Foosball Bottle Opener > Red",
            "ItemPrice": "34.95"
    }, {
        "ItemCode": "TED-072",
            "ItemQuantity": "1",
            "ItemName": "Ted Baker Womens Manicure Set",
            "ItemPrice": "74.95"
    }]
}]

6 个答案:

答案 0 :(得分:2)

我建议您使用像underscorejs / lazyjs / lodash这样的javascript库来解决这类问题。

以下是使用underscorejs的示例:

var data = [{
    "OrderID":"999123",
    "ItemCode":"TED-072",
    "ItemQuantity":"1",
    "ItemPrice":"74.95",
}, {
    "OrderID":"999123",
    "ItemCode":"DY-FBBO",
    "ItemQuantity":"2",
    "ItemName":"DOIY Foosball Bottle Opener > Red",
    "ItemPrice":"34.95",
}]

var result = _.chain(data).groupBy(function (e) {
    return e.OrderID;
}).map(function (val, key) {
    return {
        OrderID: key,
        Items: _.map(val, function (eachItem) {
            delete eachItem.OrderID;
            return eachItem;
        })
    };
}).value();

工作示例:

var data = [{
  "OrderID":"999123",
  "ItemCode":"TED-072",
  "ItemQuantity":"1",
  "ItemPrice":"74.95",
}, {
  "OrderID":"999123",
  "ItemCode":"DY-FBBO",
  "ItemQuantity":"2",
  "ItemName":"DOIY Foosball Bottle Opener > Red",
  "ItemPrice":"34.95",
}];

var result = _.chain(data).groupBy(function (e) {
  return e.OrderID;
}).map(function (val, key) {
  return {
    OrderID: key,
    Items: _.map(val, function (eachItem) {
      delete eachItem.OrderID;
      return eachItem;
    })
  };
}).value();

document.write(JSON.stringify(result));
<script src="https://cdnjs.cloudflare.com/ajax/libs/underscore.js/1.8.3/underscore-min.js"></script>

答案 1 :(得分:1)

Lodash是一个很棒的Javascript实用程序库,可以在这种情况下为您提供帮助。在您的代码中包含最新版本的lodash,并将对象分组如下:

var mergedOrders = _.groupBy(OriginalOrders, 'OrderID');

答案 2 :(得分:1)

您似乎必须执行一项功能,对于每个条目,它将检查是否匹配

试试这个:

    // your array is oldArr
var newArr = []
for (var i=0;i<oldArr.length;i++){
    var found = false;
    for(var j=0;j<newArr.length;j++){
        if(oldArr[i]["OrderID"]==newArr[j]["OrderID"]){
            newArr[j]["Items"].push(oldArr[i]);
            found=true;
            break;
    }
    if(!found){
        newArr.push({"OrderID" : oldArr[i]["OrderID"], "Items" : oldArr[i]});
    }
}

答案 3 :(得分:1)

这应该做你想做的事情,但它更像是一个群组功能而不是合并功能:)

您可以在浏览器控制台中查看结果。

var items = [
  {
    "OrderID":"999123",
    "ItemCode":"TED-072",
    "ItemQuantity":"1",
    "ItemPrice":"74.95",
  },
  {
    "OrderID":"999123",
    "ItemCode":"DY-FBBO",
    "ItemQuantity":"2",
    "ItemName":"DOIY Foosball Bottle Opener > Red",
    "ItemPrice":"34.95",
  }
];

function groupBy(ungrouped, groupByProperty) {
  var result = [], 
      getGroup = function (arr, val, groupByProperty) {
        var result, j, jlen;
        for (j = 0, jlen = arr.length; j < jlen; j++) {
          if (arr[j][groupByProperty] === val) {
            result = arr[j];
            break;
          }
        }
        if (!result) {
          result = {};
          result.items = [];
          result[groupByProperty] = val;
          arr.push(result);
        }
        return result;
      }, i, len, item;
  for (i = 0, len = ungrouped.length; i < len; i++) {
    item = getGroup(result, ungrouped[i][groupByProperty], groupByProperty);
    delete ungrouped[i][groupByProperty];
    item.items.push(ungrouped[i]);
  }
  return result;
}

var grouped = groupBy(items, 'OrderID');

document.getElementById('result').innerHTML = JSON.stringify(grouped);
console.log(grouped);
<div id="result"></div>

答案 4 :(得分:0)

您需要根据您的要求循环并创建新的分组对象。

为了更简单的方法,我建议使用jquery-linq

var qOrderIds = $.Enumerable.From(myArray).Select(function(item) { return item.OrderID; }).Distinct();
var groupedList = qOrderIds.Select(function(orderId) { 
    return { 
         OrderID: orderId, 
         Items : $.Enumerable.From(myArray).Where(function(item) { item.OrderID === orderId}).ToArray()
    };
}).ToArray();

答案 5 :(得分:0)

感谢您的所有答案。

我能够达到我的目标(可能有点脏,不像你的那么漂亮,但它在我的目的上起作用)。希望这可能在将来帮助其他人:

function processJsonObj2(dataObj, cfg) {
var retVal = dataObj.reduce(function(x, y, i, array) {

    if (x[cfg.colOrderId] === y[cfg.colOrderId]) {

        var orderId = x[cfg.colOrderId];
        var addressee = x[cfg.colAddressee];
        var company = x[cfg.colCompany];
        var addr1 = x[cfg.colAddress1];
        var addr2 = x[cfg.colAddress2];
        var suburb = x[cfg.colSuburb];
        var state = x[cfg.colState];
        var postcode = x[cfg.colPostcode];
        var country = x[cfg.colCountry];
        var orderMsg = x[cfg.colOrderMessage];
        var carrier = x[cfg.colCarrier];

        delete x[cfg.colOrderId];
        delete y[cfg.colOrderId];
        delete x[cfg.colAddressee];
        delete y[cfg.colAddressee];
        delete x[cfg.colCompany];
        delete y[cfg.colCompany];
        delete x[cfg.colAddress1];
        delete y[cfg.colAddress1];
        delete x[cfg.colAddress2];
        delete y[cfg.colAddress2];
        delete x[cfg.colSuburb];
        delete y[cfg.colSuburb];
        delete x[cfg.colState];
        delete y[cfg.colState];
        delete x[cfg.colPostcode];
        delete y[cfg.colPostcode];
        delete x[cfg.colCountry];
        delete y[cfg.colCountry];
        delete x[cfg.colOrderMessage];
        delete y[cfg.colOrderMessage];
        delete x[cfg.colCarrier];
        delete y[cfg.colCarrier];

        var orderObj = {};
        orderObj[cfg.colOrderId] = orderId;
        orderObj[cfg.colAddressee] = addressee;
        orderObj[cfg.colCompany] = company;
        orderObj[cfg.colAddress1] = addr1;
        orderObj[cfg.colAddress2] = addr2;
        orderObj[cfg.colSuburb] = suburb;
        orderObj[cfg.colState] = state;
        orderObj[cfg.colPostcode] = postcode;
        orderObj[cfg.colCountry] = country;
        orderObj[cfg.colOrderMessage] = orderMsg;
        orderObj[cfg.colCarrier] = carrier;
        orderObj["Items"] = [ x, y ];

        return orderObj;

    } else {
        var orderId = x[cfg.colOrderId];
        var addressee = x[cfg.colAddressee];
        var company = x[cfg.colCompany];
        var addr1 = x[cfg.colAddress1];
        var addr2 = x[cfg.colAddress2];
        var suburb = x[cfg.colSuburb];
        var state = x[cfg.colState];
        var postcode = x[cfg.colPostcode];
        var country = x[cfg.colCountry];
        var orderMsg = x[cfg.colOrderMessage];
        var carrier = x[cfg.colCarrier];

        var itemCode = x[cfg.colItemCode];
        var itemQuantity = x[cfg.colItemQuantity];
        var itemName = x[cfg.colItemName];
        var itemPrice = x[cfg.colitemPrice];

        var item = {};
        item[cfg.colItemCode] = itemCode;
        item[cfg.colItemQuantity] = itemQuantity;
        item[cfg.colItemName] = itemName;
        item[cfg.colItemPrice] = itemPrice;

        var orderObj = {};
        orderObj[cfg.colOrderId] = orderId;
        orderObj[cfg.colAddressee] = addressee;
        orderObj[cfg.colCompany] = company;
        orderObj[cfg.colAddress1] = addr1;
        orderObj[cfg.colAddress2] = addr2;
        orderObj[cfg.colSuburb] = suburb;
        orderObj[cfg.colState] = state;
        orderObj[cfg.colPostcode] = postcode;
        orderObj[cfg.colCountry] = country;
        orderObj[cfg.colOrderMessage] = orderMsg;
        orderObj[cfg.colCarrier] = carrier;
        orderObj["Items"] = [ item ];

        return orderObj;
    }

});

return retVal;

}