Rac*_*hel 6 arrays bash shell command-line
我看了,但只看到了一个脚本中传递的数组的答案.
我想将多个数组传递给bash脚本,该脚本将它们分配为单个变量,如下所示:
./myScript.sh ${array1[@]} ${array2[@]} ${array3[@]}
Run Code Online (Sandbox Code Playgroud)
这样的:var1=array1和var2=array2和var3=array3
我已经尝试了多个选项,但是variableName=("$@")将所有数组组合成每个变量.我希望在我的bash脚本中有一个代表每个数组的变量.
shell将一个参数向量(也就是说,一个简单的C字符串数组)传递给正在运行的程序.这是操作系统级别的限制:没有方法可以在参数列表中的两个程序(任何两个程序,用任何语言编写!)之间传递结构化数据,除非通过在该数组的成员的内容中编码该结构C字符串.
如果效率是一个目标(无论是在解析的容易程度还是在ARG_MAX命令行和环境存储的限制之外使用的空间量),要考虑的一种方法是在每个数组前面加上描述其长度的参数.
但是,通过提供长度参数,您可以指示该参数列表的哪些部分应该是给定数组的一部分:
./myScript \
"${#array1[@]}" "${array1[@]}" \
"${#array2[@]}" "${array2[@]}" \
"${#array3[@]}" "${array3[@]}"
Run Code Online (Sandbox Code Playgroud)
...然后,在脚本中,您可以使用length参数将内容拆分回数组:
#!/usr/bin/env bash
array1=( "${@:2:$1}" ); shift "$(( $1 + 1 ))"
array2=( "${@:2:$1}" ); shift "$(( $1 + 1 ))"
array3=( "${@:2:$1}" ); shift "$(( $1 + 1 ))"
declare -p array1 array2 array3
Run Code Online (Sandbox Code Playgroud)
如果运行为./myScript 3 a b c 2 X Y 1 z,则具有输出:
declare -a array1='([0]="a" [1]="b" [2]="c")'
declare -a array2='([0]="X" [1]="Y")'
declare -a array3='([0]="z")'
Run Code Online (Sandbox Code Playgroud)
顺便提一下,Python世界中常见的做法(特别是对于argparse库的用户)是允许多次传递参数来修改给定数组.在shell中,这看起来像:
./myScript \
"${array1[@]/#/--array1=}" \
"${array2[@]/#/--array2=}" \
"${array3[@]/#/--array3=}"
Run Code Online (Sandbox Code Playgroud)
然后解析它的代码可能如下所示:
#!/usr/bin/env bash
declare -a args array1 array2 array3
while (( $# )); do
case $1 in
--array1=*) array1+=( "${1#*=}" );;
--array2=*) array2+=( "${1#*=}" );;
--array3=*) array3+=( "${1#*=}" );;
*) args+=( "$1" );;
esac
shift
done
Run Code Online (Sandbox Code Playgroud)
因此,如果您的原始值是array1=( one two three ) array2=( aye bee ) array3=( "hello world" ),则调用约定为:
./myScript --array1=one --array1=two --array1=three \
--array2=aye --array2=bee \
--array3="hello world"
Run Code Online (Sandbox Code Playgroud)
另一种方法是为每个数组传递一个文件名,从中可以读取NUL分隔的内容列表.这种方法的一个主要优点是数组内容的大小不计入ARG_MAX操作系统强制命令行长度限制.此外,对于可用的操作系统,下面不会创建真正的磁盘文件,而是创建/dev/fd由写入每个数组内容的子shell写入的FIFO的样式链接.
./myScript \
<( (( ${#array1[@]} )) && printf '%s\0' "${array1[@]}") \
<( (( ${#array2[@]} )) && printf '%s\0' "${array2[@]}") \
<( (( ${#array3[@]} )) && printf '%s\0' "${array3[@]}")
Run Code Online (Sandbox Code Playgroud)
...阅读(使用bash 4.4或更新版本提供mapfile -d):
#!/usr/bin/env bash
mapfile -d '' array1 <"$1"
mapfile -d '' array2 <"$2"
mapfile -d '' array3 <"$3"
Run Code Online (Sandbox Code Playgroud)
...或者,支持旧的bash版本:
#!/usr/bin/env bash
declare -a array1 array2 array3
while IFS= read -r -d '' entry; do array1+=( "$entry" ); done <"$1"
while IFS= read -r -d '' entry; do array2+=( "$entry" ); done <"$2"
while IFS= read -r -d '' entry; do array3+=( "$entry" ); done <"$3"
Run Code Online (Sandbox Code Playgroud)