如何使用 jq 将 json 文档中的空值替换为特定值?
How can I replace a null value in a json document by a specific value with jq?
我有一个 json 文件,其中包含 null
s 作为某些键的值,我想将其替换为一些特定值。
鉴于此输入:
{
"id": null,
"rows": [
{
"panels": [
{
"title": "Panel1",
"datasource": null
},
{
"title": "Panel2",
"datasource": null
}
]
}
]
}
我想要
{
"id": null,
"rows": [
{
"panels": [
{
"title": "Panel1",
"datasource": "mydb"
},
{
"title": "Panel2",
"datasource": "mydb"
}
]
}
]
}
我目前使用的是
sed 's/"datasource": null/"datasource": "mydb"/'
这会产生我需要的输出,但我一直认为使用 sed
来完成这项工作是一种耻辱,因为有像 jq
这样的工具可以在 [=28= 上工作] 以更好的方式。
首先您需要确定要更新的对象。由于您想将面板的空数据源设置为 "mydb"
,您可以这样做:
$ jq '.rows[].panels[].datasource //= "mydb"' input.json
如果您想在任何级别的任何对象上更新任何 datasource
属性,您可以使用 ..
递归搜索它们。
$ jq '(.. | select(objects | has("datasource"))).datasource //= "mydb"' input.json
郑重声明,将所有为空的键值(无论它们出现在何处)更改为其他值也很容易。 jq > 1.5 的版本包括一个 jq 定义的过滤器,walk/1,它也可以在其他版本的 jq 中使用(例如通过复制和粘贴)。
例如,要使用 walk/1 将所有为空的键值更改为 0:
walk(if type == "object" then with_entries(.value //= 0) else . end)
这里是jq提供的walk/1定义:
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;
这是一个解决方案,它使用 tostream、reduce 和 setpath 来设置任何叶子datasource
属性值为 null
到 "mydb"
.
reduce (tostream|select(length==2)) as [$p,$v] (
.
; if $p[-1] == "datasource" and $v == null then setpath($p; "mydb") else . end
)
我有一个 json 文件,其中包含 null
s 作为某些键的值,我想将其替换为一些特定值。
鉴于此输入:
{
"id": null,
"rows": [
{
"panels": [
{
"title": "Panel1",
"datasource": null
},
{
"title": "Panel2",
"datasource": null
}
]
}
]
}
我想要
{
"id": null,
"rows": [
{
"panels": [
{
"title": "Panel1",
"datasource": "mydb"
},
{
"title": "Panel2",
"datasource": "mydb"
}
]
}
]
}
我目前使用的是
sed 's/"datasource": null/"datasource": "mydb"/'
这会产生我需要的输出,但我一直认为使用 sed
来完成这项工作是一种耻辱,因为有像 jq
这样的工具可以在 [=28= 上工作] 以更好的方式。
首先您需要确定要更新的对象。由于您想将面板的空数据源设置为 "mydb"
,您可以这样做:
$ jq '.rows[].panels[].datasource //= "mydb"' input.json
如果您想在任何级别的任何对象上更新任何 datasource
属性,您可以使用 ..
递归搜索它们。
$ jq '(.. | select(objects | has("datasource"))).datasource //= "mydb"' input.json
郑重声明,将所有为空的键值(无论它们出现在何处)更改为其他值也很容易。 jq > 1.5 的版本包括一个 jq 定义的过滤器,walk/1,它也可以在其他版本的 jq 中使用(例如通过复制和粘贴)。
例如,要使用 walk/1 将所有为空的键值更改为 0:
walk(if type == "object" then with_entries(.value //= 0) else . end)
这里是jq提供的walk/1定义:
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;
这是一个解决方案,它使用 tostream、reduce 和 setpath 来设置任何叶子datasource
属性值为 null
到 "mydb"
.
reduce (tostream|select(length==2)) as [$p,$v] (
.
; if $p[-1] == "datasource" and $v == null then setpath($p; "mydb") else . end
)