从bash关联数组构造json哈希

hta*_*ess 6 bash json jq

我想将bash中的关联数组转换为json hash/dict.我更喜欢使用jq来做这个,因为它已经是一个依赖项,我可以依靠它来生成格式良好的json.有人可以演示如何实现这一目标吗?

#!/bin/bash

declare -A dict=()

dict["foo"]=1
dict["bar"]=2
dict["baz"]=3

for i in "${!dict[@]}"
do
    echo "key  : $i"
    echo "value: ${dict[$i]}"
done

echo 'desired output using jq: { "foo": 1, "bar": 2, "baz": 3 }'
Run Code Online (Sandbox Code Playgroud)

pea*_*eak 6

有很多可能性,但鉴于您已经编写了一个bash for循环,您可能希望从脚本的这种变体开始:

#!/bin/bash
# Requires bash with associative arrays
declare -A dict

dict["foo"]=1
dict["bar"]=2
dict["baz"]=3

for i in "${!dict[@]}"
do
    echo "$i" 
    echo "${dict[$i]}"
done |
jq -n -R 'reduce inputs as $i ({}; . + { ($i): (input|(tonumber? // .)) })'
Run Code Online (Sandbox Code Playgroud)

结果反映了bash for循环生成的键的顺序:

{
  "bar": 2,
  "baz": 3,
  "foo": 1
}
Run Code Online (Sandbox Code Playgroud)

一般来说,基于jq键值对的方法,在一行上有一个键后跟下一行的相应值,有很多建议.遵循该一般方案的通用解决方案,但使用NUL作为"线端"字符,在下面给出.

键和值作为JSON实体

为了使上述更通用,最好将键和值呈现为JSON实体.在本案例中,我们可以写:

for i in "${!dict[@]}"
do
    echo "\"$i\""
    echo "${dict[$i]}"
done | 
jq -n 'reduce inputs as $i ({}; . + { ($i): input })'
Run Code Online (Sandbox Code Playgroud)

其他变化

JSON键必须是JSON字符串,因此可能需要一些工作来确保实现从bash键到JSON键的所需映射.类似的注释适用于从bash数组值到JSON值的映射.处理任意bash键的一种方法是让jq进行转换:

printf "%s" "$i" | jq -Rs .
Run Code Online (Sandbox Code Playgroud)

您当然可以使用bash数组值执行相同的操作,并让jq检查该值是否可以根据需要转换为数字或其他JSON类型(例如,使用fromjson? // .).

通用解决方案

这是一个通用的解决方案,沿着jq常见问题解答中提到并由@CharlesDuffy提倡.它在将bash键和值传递给jq时使用NUL作为分隔符,并且具有仅需要一次调用jq的优点.如果需要,fromjson? // .可以省略过滤器或用另一个过滤器替换过滤器.

declare -A dict=( [$'foo\naha']=$'a\nb' [bar]=2 [baz]=$'{"x":0}' )

for key in "${!dict[@]}"; do
    printf '%s\0%s\0' "$key" "${dict[$key]}"
done |
jq -Rs '
  split("\u0000")
  | . as $a
  | reduce range(0; length/2) as $i 
      ({}; . + {($a[2*$i]): ($a[2*$i + 1]|fromjson? // .)})'
Run Code Online (Sandbox Code Playgroud)

输出:

{
  "foo\naha": "a\nb",
  "bar": 2,
  "baz": {
    "x": 0
  }
}
Run Code Online (Sandbox Code Playgroud)


Ber*_*tel 5

您可以将变量初始化为空对象{},并为每次迭代添加键/值{($key):$value},将结果重新注入到同一变量中:

#!/bin/bash

declare -A dict=()

dict["foo"]=1
dict["bar"]=2
dict["baz"]=3

data='{}'

for i in "${!dict[@]}"
do
    data=$(jq -n --arg data "$data" \
                 --arg key "$i"     \
                 --arg value "${dict[$i]}" \
                 '$data | fromjson + { ($key) : ($value | tonumber) }')
done

echo "$data"
Run Code Online (Sandbox Code Playgroud)

  • 很好的解决方案,我喜欢数据作为 jq arg 传递并迭代扩展的方式。 (2认同)

hta*_*ess 5

这个答案nico103来自freenode #jq

#!/bin/bash

declare -A dict=()

dict["foo"]=1
dict["bar"]=2
dict["baz"]=3

assoc2json() {
    declare -n v=$1
    printf '%s\0' "${!v[@]}" "${v[@]}" |
    jq -Rs 'split("\u0000") | . as $v | (length / 2) as $n | reduce range($n) as $idx ({}; .[$v[$idx]]=$v[$idx+$n])'
}

assoc2json dict
Run Code Online (Sandbox Code Playgroud)