在脚本中我有一个关联数组,例如:
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]}'" )
对代码的这些更改中的任何一个都将在选项值中包含单引号,因此对值的重新评估将变为必要的。