递归列出JSON中的所有对象属性路径

Recursively list all object property paths from JSON

本文关键字:对象 属性 路径 JSON 递归      更新时间:2023-09-26

我必须与JSON返回(这是我从Amazon Marketplace API的XML响应中得到的)作斗争。它包含数组、子数组等等。目前,我只需要获得2个值(AmazonOrderId和OrderItem对象),但我找不到直接解决它们的方法。我仍然会尝试几个小时。递归的东西让我有点害怕。

有那么多的工具,也用于美化或编辑JSON。有没有可能有一个解决方案递归地显示所有的属性链接?到目前为止,我还没有找到这样的东西。

UPDATE:从斜眼的答案的代码,我已经建立了一个快速和肮脏的工具,做我想要的。网址:http://output.jsbin.com/gunixo

:

myJSON.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].QuantityOrdered
myJSON.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].Title
...

在这种情况下,这会节省时间。

    {
    "ListOrderItemsResponse": {
        "$": {
            "xmlns": "https://mws.amazonservices.com/Orders/2013-09-01"
        },
        "ListOrderItemsResult": [{
            "OrderItems": [{
                "OrderItem": [{
                    "QuantityOrdered": ["1"],
                    "Title": ["My Product Title"],
                    "ShippingTax": [{
                        "CurrencyCode": ["EUR"],
                        "Amount": ["0.00"]
                    }],
                    "PromotionDiscount": [{
                        "CurrencyCode": ["EUR"],
                        "Amount": ["0.00"]
                    }],
                    "ConditionId": ["New"],
                    "ASIN": ["BXXXI5W6M8"],
                    "SellerSKU": ["KNDR-BTTWSCH-BGSBNNY"],
                    "OrderItemId": ["67050234000659"],
                    "GiftWrapTax": [{
                        "CurrencyCode": ["EUR"],
                        "Amount": ["0.00"]
                    }],
                    "QuantityShipped": ["1"],
                    "ShippingPrice": [{
                        "CurrencyCode": ["EUR"],
                        "Amount": ["4.90"]
                    }],
                    "GiftWrapPrice": [{
                        "CurrencyCode": ["EUR"],
                        "Amount": ["0.00"]
                    }],
                    "ConditionSubtypeId": ["New"],
                    "ItemPrice": [{
                        "CurrencyCode": ["EUR"],
                        "Amount": ["15.77"]
                    }],
                    "ItemTax": [{
                        "CurrencyCode": ["EUR"],
                        "Amount": ["0.00"]
                    }],
                    "ShippingDiscount": [{
                        "CurrencyCode": ["EUR"],
                        "Amount": ["0.00"]
                    }]
                }]
            }],
            "AmazonOrderId": ["304-6107803-0000000"]
        }],
        "ResponseMetadata": [{
            "RequestId": ["dd64dd40-d939-4ea1-875d-xxxxxxxxxxxx"]
        }]
    }
}

如果您确实只需要这两个值,您可以在解析时执行。

var data = {};
// ----------v--- Original JSON data before parsing.
JSON.parse(myJSON, function(k, v) {
    if (k === "AmazonOrderId" || k === "OrderItem") {
        data[k] = v;
    }
    return v;
});

说……递归的东西不应该可怕。在StackOverflow上有许多预先编写的函数接受你正在寻找的属性并返回值,所以你可以使用其中一个。


这是一个简单的递归函数,它生成所有数据的路径。

function objectToPaths(data) {
  var validId = /^[a-z_$][a-z0-9_$]*$/i;
  var result = [];
  doIt(data, "");
  return result;
  function doIt(data, s) {
    if (data && typeof data === "object") {
      if (Array.isArray(data)) {
        for (var i = 0; i < data.length; i++) {
          doIt(data[i], s + "[" + i + "]");
        }
      } else {
        for (var p in data) {
          if (validId.test(p)) {
            doIt(data[p], s + "." + p);
          } else {
            doIt(data[p], s + "['"" + p + "'"]");
          }
        }
      }
    } else {
      result.push(s);
    }
  }
}

单独列出所有Array成员。如果您知道这些数组成员中的数据结构总是相同的,则可以整合这一点。

您的数据的结果:

.ListOrderItemsResponse.$.xmlns
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].QuantityOrdered[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].Title[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ShippingTax[0].CurrencyCode[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ShippingTax[0].Amount[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].PromotionDiscount[0].CurrencyCode[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].PromotionDiscount[0].Amount[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ConditionId[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ASIN[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].SellerSKU[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].OrderItemId[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].GiftWrapTax[0].CurrencyCode[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].GiftWrapTax[0].Amount[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].QuantityShipped[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ShippingPrice[0].CurrencyCode[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ShippingPrice[0].Amount[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].GiftWrapPrice[0].CurrencyCode[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].GiftWrapPrice[0].Amount[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ConditionSubtypeId[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ItemPrice[0].CurrencyCode[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ItemPrice[0].Amount[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ItemTax[0].CurrencyCode[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ItemTax[0].Amount[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ShippingDiscount[0].CurrencyCode[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].OrderItems[0].OrderItem[0].ShippingDiscount[0].Amount[0]
.ListOrderItemsResponse.ListOrderItemsResult[0].AmazonOrderId[0]
.ListOrderItemsResponse.ResponseMetadata[0].RequestId[0]