JQ:从多个数组中删除对象
JQ: Remove object from multiple arrays
我想使用 jq 从输入数据的所有数组中删除具有给定名称的所有对象。例如从中删除 "Name1":
{
"Category1": [
{
"name": "Name1",
"desc": "Desc1"
},
{
"name": "Name2",
"desc": "Desc2"
}
],
"Category2": [
{
"name": "Name1",
"desc": "Desc1"
},
{
"name": "Name3",
"desc": "Desc3"
}
],
"Category3": [
{
"name": "Name4",
"desc": "Desc4"
}
]
}
应该产生这个:
{
"Category1": [
{
"name": "Name2",
"desc": "Desc2"
}
],
"Category2": [
{
"name": "Name3",
"desc": "Desc3"
}
],
"Category3": [
{
"name": "Name4",
"desc": "Desc4"
}
]
}
我没有使用过 jq,甚至 JSON,很多,经过几个小时的谷歌搜索和试验后,我还没有弄明白。我该怎么做?
我最接近的是这个:
cat input | jq 'keys[] as $k | .[$k] |= map( select( .name != "Name1"))'
这确实过滤了每个数组,但 returns 结果作为三个单独的对象,这不是我想要的。
这是一个解决方案,可以删除所有具有指定名称的对象,无论它们出现在哪里。它使用通用函数 walk/1,
这是 jq > 1.5 的内置版本,因此如果您的 jq 包含它可以省略,但是冗余地包含它没有坏处,例如在 jq 脚本中。
# Apply f to composite entities recursively, and to atoms
def walk(f):
. as $in
| if type == "object" then
reduce keys[] as $key
( {}; . + { ($key): ($in[$key] | walk(f)) } ) | f
elif type == "array" then map( walk(f) ) | f
else f
end;
walk(if type == "object" and .name == "Name1" then empty else . end)
如果你真的只想从数组中删除对象,那么你可以使用:
walk(if type == "array" then map(select( type != "object" or .name != "Name1")) else . end)
如果您输入的结构 JSON 始终如示例所示,请尝试以下操作:
map_values(map(select(.name != "Name1")))
这是一个使用 reduce 和 del
的解决方案
reduce keys[] as $k (
.
; del(.[$k][] | select(.name == "Name1"))
)
我想使用 jq 从输入数据的所有数组中删除具有给定名称的所有对象。例如从中删除 "Name1":
{
"Category1": [
{
"name": "Name1",
"desc": "Desc1"
},
{
"name": "Name2",
"desc": "Desc2"
}
],
"Category2": [
{
"name": "Name1",
"desc": "Desc1"
},
{
"name": "Name3",
"desc": "Desc3"
}
],
"Category3": [
{
"name": "Name4",
"desc": "Desc4"
}
]
}
应该产生这个:
{
"Category1": [
{
"name": "Name2",
"desc": "Desc2"
}
],
"Category2": [
{
"name": "Name3",
"desc": "Desc3"
}
],
"Category3": [
{
"name": "Name4",
"desc": "Desc4"
}
]
}
我没有使用过 jq,甚至 JSON,很多,经过几个小时的谷歌搜索和试验后,我还没有弄明白。我该怎么做?
我最接近的是这个:
cat input | jq 'keys[] as $k | .[$k] |= map( select( .name != "Name1"))'
这确实过滤了每个数组,但 returns 结果作为三个单独的对象,这不是我想要的。
这是一个解决方案,可以删除所有具有指定名称的对象,无论它们出现在哪里。它使用通用函数 walk/1, 这是 jq > 1.5 的内置版本,因此如果您的 jq 包含它可以省略,但是冗余地包含它没有坏处,例如在 jq 脚本中。
# Apply f to composite entities recursively, and to atoms
def walk(f):
. as $in
| if type == "object" then
reduce keys[] as $key
( {}; . + { ($key): ($in[$key] | walk(f)) } ) | f
elif type == "array" then map( walk(f) ) | f
else f
end;
walk(if type == "object" and .name == "Name1" then empty else . end)
如果你真的只想从数组中删除对象,那么你可以使用:
walk(if type == "array" then map(select( type != "object" or .name != "Name1")) else . end)
如果您输入的结构 JSON 始终如示例所示,请尝试以下操作:
map_values(map(select(.name != "Name1")))
这是一个使用 reduce 和 del
的解决方案reduce keys[] as $k (
.
; del(.[$k][] | select(.name == "Name1"))
)