在脚本中,我有一个关联数组,例如:
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
使用辅助功能:
上述脚本中的最终命令将扩展为相当于编写
该
to_param_list
函数采用数组变量的名称和关联数组变量的名称,并使用它们在函数中创建两个“名称引用”变量(4.3 版中引入了名称引用)bash
。这些然后用于使用关联数组中适当格式的键和值填充给定的数组变量。函数中的循环遍历
"${!inhash[@]}"
,这是关联数组中单独引用的键的列表。一旦函数调用返回,脚本将使用该数组来调用您的其他脚本或命令。
运行上面的
脚本将输出
这表明选项是在没有分词或文件名通配生效的情况下生成的。它还表明,键的顺序可能不会被保留,因为从关联数组中访问键将以相当随机的顺序进行。
您不能在这里真正安全地使用命令替换,因为它的结果将是一个字符串。如果不加引号,则此字符串将在空白字符上拆分(默认情况下),这将另外拆分关联数组的键和值。shell 还会对生成的单词执行文件名通配。双引号命令替换将无济于事,因为这会导致您
my_script.sh
使用单个参数调用您。关于您的问题
makeself
:该
makeself
脚本使用安装程序脚本的参数执行此操作:这会将参数保存为字符串
$SCRIPTARGS
(连接,用空格分隔)。这稍后会按原样插入到自解压存档中。为了在重新评估选项时正确解析选项(在运行安装程序时),您必须在参数值中提供一组额外的引号,以便正确分隔它们。请注意,这不是我的代码中的错误。这只是根据用户提供的值
makeself
生成shell 代码的副作用。理想情况下,
makeself
脚本应该编写每个提供的参数,并在它们周围加上一组额外的引号,但事实并非如此,因为很难知道它可能会产生什么影响。相反,它让用户提供这些额外的报价。从上面重新运行我的测试,但现在
生产
您可以看到,当 shell重新评估这些字符串时,它们不会被空格分割。
显然,您可以使用初始关联数组,而是
to_param_list
通过更改在函数中添加引号进入
对代码的这些更改中的任何一个都将在选项值中包含单引号,因此需要重新评估这些值。