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

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

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

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

是否有一个命令可以将其转换为以下形式的参数列表

--key1=value1 --key2=value2

无需手动重写

 --key1="${VARS[key1]}" --key2="${VARS[key2]}"

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

my_script.sh $(to_param_list $VARS)

为了扩展我对 @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[@]}"

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

while ! -z "$1" ; do
    echo "$1"
    shift
done

并传递一个数组,例如:

installer_param_array=( ["upgrade-from"]="19 .2.0" ["upgrade-to"]="19.3.0" )

结果是这样的输出:

--upgrade-to=19.3.0
--upgrade-from=19
.2.0

答案1

具有辅助功能:

#!/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[@]}"

上述脚本中的最后一个命令将扩展为相当于编写

my_script.sh "--key2=value" "--key1=value1"

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[@]}"

脚本会输出

Arg: --key2=some thing
Arg: --key3=* * *
Arg: --key1=hello world

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


您实际上不能在这里安全地使用命令替换,因为它的结果将是单个字符串。如果不加引号,则该字符串将在空白字符上拆分(默认情况下),这还会拆分关联数组的键和值。 shell 还会对结果单词执行文件名通配。双引号命令替换不会有帮助,因为这会导致my_script.sh使用单身的争论。


关于您的问题makeself:

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

SCRIPTARGS="$*"

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

installer_param_array=( ["upgrade-from"]="'19 .2.0'" ["upgrade-to"]="'19.3.0'" )

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

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

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

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

to_param_list list my_vars
printf 'Arg: %s\n' "${list[@]}"

产生

Arg: --key2='some value'
Arg: --key3='* * *'
Arg: --key1='hello world'

你可以看到这些字符串,当重新评估通过外壳,不会因空格而分裂。

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

outlist+=( "--$param=${inhash[$param]}" )

进入

outlist+=( "--$param='${inhash[$param]}'" )

对代码的这些更改中的任何一个都将在选项值中包含单引号,因此对值的重新评估将变为必要的

相关内容