将多个数组作为参数传递给Bash脚本?

Rac*_*hel 6 arrays bash shell command-line

我看了,但只看到了一个脚本中传递的数组的答案.

我想将多个数组传递给bash脚本,该脚本将它们分配为单个变量,如下所示:

./myScript.sh ${array1[@]} ${array2[@]} ${array3[@]}
Run Code Online (Sandbox Code Playgroud)

这样的:var1=array1var2=array2var3=array3

我已经尝试了多个选项,但是variableName=("$@")将所有数组组合成每个变量.我希望在我的bash脚本中有一个代表每个数组的变量.

Cha*_*ffy 9

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划分的流

另一种方法是为每个数组传递一个文件名,从中可以读取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)