将关联数组作为参数列表传递给脚本

Mat*_*ari 10 bash parameter associative-array

在脚本中,我有一个关联数组,例如:

declare -A VARS=( ["key1"]="value1" ["key2"]="value" )
Run Code Online (Sandbox Code Playgroud)

是否有一个命令可以将其转换为表单中的参数列表

--key1=value1 --key2=value2
Run Code Online (Sandbox Code Playgroud)

无需手动重写

 --key1="${VARS[key1]}" --key2="${VARS[key2]}"
Run Code Online (Sandbox Code Playgroud)

我想到的用例是将数组作为参数列表传递给脚本,例如

my_script.sh $(to_param_list $VARS)
Run Code Online (Sandbox Code Playgroud)

为了扩展我对@Kusalananda 答案所做的评论,我的确切用例如下:我有一个脚本,用于使用 makeself 构建自解压安装程序,并且该脚本接收一些要分开的参数:

  • 脚本本身的参数
  • 自解压安装程序中安装程序的参数

然后脚本会像这样构建安装程序:

to_param_list installer_param_list installer_param_array
./makeself ./path/to/sourcedir ./path/to/created/installer "My installer" ./path/to/install/inside/package "${installer_param_list[@]}"
Run Code Online (Sandbox Code Playgroud)

但是,我已经使用包内的一个非常简单的安装程序脚本测试了参数传递:

while ! -z "$1" ; do
    echo "$1"
    shift
done
Run Code Online (Sandbox Code Playgroud)

并传递一个数组,如:

installer_param_array=( ["upgrade-from"]="19 .2.0" ["upgrade-to"]="19.3.0" )
Run Code Online (Sandbox Code Playgroud)

导致此输出:

--upgrade-to=19.3.0
--upgrade-from=19
.2.0
Run Code Online (Sandbox Code Playgroud)

Kus*_*nda 14

使用辅助函数:

#!/bin/bash

to_param_list () {
    declare -n outlist=$1
    declare -n inhash=$2

    for param in "${!inhash[@]}"; do
        outlist+=( "--$param=${inhash[$param]}" )
    done
}

declare -A my_vars=( ["key1"]="value1" ["key2"]="value" )

to_param_list list my_vars
my_script.sh "${list[@]}"
Run Code Online (Sandbox Code Playgroud)

上面脚本中的最后一个命令将扩展为等价于编写

my_script.sh "--key2=value" "--key1=value1"
Run Code Online (Sandbox Code Playgroud)

to_param_list函数采用名称的数组变量和所述名称关联数组变量,并且使用这些来创建在函数两个“名称参考”变量(namerefs是在引入bash释放4.3)。然后,这些用于使用关联数组中适当格式的键和值填充给定的数组变量。

函数中的循环迭代"${!inhash[@]}",它是关联数组中单独引用的键的列表。

一旦函数调用返回,脚本将使用该数组来调用您的其他脚本或命令。

运行上面的

declare -A my_vars=( ["key1"]="hello world" ["key2"]="some thing" ["key3"]="* * *" )

to_param_list list my_vars
printf 'Arg: %s\n' "${list[@]}"
Run Code Online (Sandbox Code Playgroud)

脚本将输出

Arg: --key2=some thing
Arg: --key3=* * *
Arg: --key1=hello world
Run Code Online (Sandbox Code Playgroud)

这表明选项是在没有分词或文件名通配生效的情况下生成的。它还表明键的顺序可能不会被保留,因为从关联数组中访问键将以相当随机的顺序进行。


您不能真正在这里安全地使用命令替换,因为它的结果将是单个字符串。如果未加引号,则此字符串将被拆分为空白字符(默认情况下),这将另外拆分关联数组的键和值。shell 还会对结果单词执行文件名通配。双引号命令替换无济于事,因为这会导致my_script.sh使用单个参数调用您。


关于您的问题makeself

makeself脚本使用安装程序脚本的参数执行此操作:

SCRIPTARGS="$*"
Run Code Online (Sandbox Code Playgroud)

这将参数保存为字符串$SCRIPTARGS(连接,用空格分隔)。这稍后会按原样插入到自解压存档中。为了在重新评估时正确解析选项(它们在运行安装程序时),您必须在参数值中提供一组额外的引号,以便正确分隔它们。

installer_param_array=( ["upgrade-from"]="'19 .2.0'" ["upgrade-to"]="'19.3.0'" )
Run Code Online (Sandbox Code Playgroud)

请注意,这不是我代码中的错误。这只是基于用户提供的值makeself生成shell 代码的副作用。

理想情况下,makeself脚本应该编写每个提供的参数,并在它们周围加上一组额外的引号,但可能没有,因为很难知道可能会产生什么影响。相反,它留给用户提供这些额外的报价。

从上面重新运行我的测试,但现在

declare -A my_vars=( ["key1"]="'hello world'" ["key2"]="'some value'" ["key3"]="'* * *'" )

to_param_list list my_vars
printf 'Arg: %s\n' "${list[@]}"
Run Code Online (Sandbox Code Playgroud)

产生

Arg: --key2='some value'
Arg: --key3='* * *'
Arg: --key1='hello world'
Run Code Online (Sandbox Code Playgroud)

您可以看到,这些字符串在由 shell重新评估时不会在空格上拆分。

显然,您可以使用初始关联数组,而是to_param_list通过更改在函数中添加引号

outlist+=( "--$param=${inhash[$param]}" )
Run Code Online (Sandbox Code Playgroud)

进入

outlist+=( "--$param='${inhash[$param]}'" )
Run Code Online (Sandbox Code Playgroud)

无论这些变化对你的代码将包括选项的数值单引号,所以值的重新评估将成为必要