JQ如何将多个对象合并为一个

时间:2018-07-24 05:49:48

标签: json object stream jq jsonstream

给出以下输入(这是具有另一个复杂查询的100K +个对象的输出的精简版本):

echo '{ "a": { "b":"c", "d":"e" } }{ "a": { "b":"f", "d":"g" } }' | jq '.'
{
  "a": {
    "b": "c",
    "d": "e"
  }
}
{
  "a": {
    "b": "f",
    "d": "g"
  }
}

所需的输出:

{
   "c": "e",
   "f": "g"
}

或(更适合后续使用):

{
   x: {
      "c": "e",
      "f": "g"
   }
}

我无法为自己的一生弄清楚该怎么做。我真正的问题当然是多对象输入数据,对此我真的不知道它是否是有效的JSON。 Jq会产生并接受它,而jshon不会。我尝试了各种可能性,但是没有一种能满足我的需求。我认为这是最有可能的候选人:

echo '{ "a": { "b":"c", "d":"e" } }{ "a": { "b":"f", "d":"g" } }' | jq ' . | { (.a.b): .a.d }'
{
   "c": "e"
}
{
   "f": "g"
}

但是a。我尝试过的其他事情:

' . | { x: { (.a.b): .a.d } }'
'{ x: {} | . | add }'
'{ x: {} | . | x += }'
'{ x: {} | x += . }'
'x: {} | .x += { (.a.b): .a.d }'
'{ x: {} } | .x += { (.a.b): .a.d }'

另一个,关闭,但没有提示音

'reduce { (.a.b): .a.d } as $item ({}; . + $item)'
{
  "c": "e"
}
{
  "f": "g"
}

谁愿意启发我?

因此,感谢@peak,上述用例的完整答案是

echo '{ "a": { "b": "c", "d": "e" } }{ "a": { "b": "f", "d": "g" } }' | jq -n '{ x: [inputs | .a | { (.b): .d} ] | add }'
{
  "x": {
    "c": "e",
    "f": "g"
  }
}

1 个答案:

答案 0 :(得分:2)

让我们假设您拥有jq 1.5或更高版本,并且JSON对象位于一个或多个文件中。然后,对于您的第一个预期输出,您可以简单地编写:

[inputs | .a | { (.b): .d} ] | add

这假定您使用-n命令行选项。我敢肯定,一旦您看到解决方案有多简单,便无需赘述。

对于第二个预期的输出,只需将上面的行包装在:

{x: _}

例如:

$ jq -ncf program.jq input.json
{"x":{"c":"e","f":"g"}}

p.s。。使用reduce的方法很好,但是您需要使用-s命令行选项。

p.p.s。。您还欠我一杯啤酒吗?