使用通过命令行传递的路径访问 JQ 中的嵌套值

问题描述

如何或是否可以从指定路径作为命令参数的 JSON 结构中提取数据。为了简单起见,我从一些较大的脚本中获取了这个简单的片段,但在解决过程中遇到了问题:

#!/bin/bash

DATA='{
        "level1": {
                "level2": {
                        "level3": {
                                "foo": "bar","bar": "baz","baz": "bar"
                        }
                }
        }

}'

field="level1.level2.level3"

# does not work
jq -r --arg f ${field} '.[$f] | to_entries | .[] | "\"" + .key + "\"=\"" + .value + "\""' <<< ${DATA}

# works
jq -r --arg f ${field} '.level1.level2.level3 | to_entries | .[] | "\"" + .key + "\"=\"" + .value + "\""' <<< ${DATA}

# also works                              
field2="level3"

jq -r --arg f ${field2} '.level1.level2 | .[$f] | to_entries | .[] | "\"" + .key + "\"=\"" + .value + "\""' <<< ${DATA}

给出以下输出

user@astra:~/test$ ./jqtest 
jq: error (at <stdin>:12): null (null) has no keys
"foo"="bar"
"bar"="baz"
"baz"="bar"
"foo"="bar"
"bar"="baz"
"baz"="bar"

我做错了什么?

解决方法

在这种情况下,.[$f] 表示“返回与名为 level1.level2.level3 的键关联的值”。见:

$ jq --arg f 'level1.level2.level3' '.[$f]' <<< '{ "level1.level2.level3": "foo" }'
"foo"

除非任何路径组件包含一个点,否则用点分割 $f 并将结果用作 getpath 的参数应该可以工作。

getpath($f / ".")
,

jq 似乎对输入参数中的路径很挑剔。

一种解决方案是将路径作为 json 数组提供,然后使用 getpath 将其转换为路径:

field='["level1","level2","level3"]'
jq -r --argjson f "$field" 'getpath($f)' <<< ${DATA}

或者针对您的具体问题:


field='["level1","level3"]'

jq -r --argjson f "${field}" 'getpath($f) | to_entries | .[] | "\"" + .key + "\"=\"" + .value + "\""' <<< ${DATA}

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...