1

我有一个源文件,其中包含用 Bash 编写的代码。源文件中定义了一些函数,例如

 - abc(i,j,k)

我需要使用 shell 脚本读取源文件,并且需要生成所有可能的函数调用组合,例如:

输出

 - abc(i,j,k)
 - abc(i,j,)
 - abc(i,,j)
 - abc(,j,k)
 - abc(,,k)
 - abc(i,,)
 - abc(,j,)
 - abc(,,)

对于三个参数,有 8 种组合,因此组合将根据参数的数量增加或减少。

4

2 回答 2

4

bash 的大括号展开可以生成组合:

printf -- "- abc(%s}\n" {i,},{j,},{k,}

但我假设你希望它比这更有活力。这是看起来很难看的代码,但它可以工作。

while IFS= read -r line; do
    # match a line that looks like a function,
    # and capture the function name and the parameter list
    if [[ $line =~ " - "([[:alnum:]]+)\(([a-z,]+)\) ]]; then
        funcname=${BASH_REMATCH[1]}
        # split the comma-separated parameters into an array
        IFS=, read -ra params <<< "${BASH_REMATCH[2]}"
        # create the brace-expansion string
        printf -v combos "{%s,}," "${params[@]}"
        # and this is the command to print out the combinations
        cmd=( printf -- "\"- $funcname(%s)\n\"" ${combos%,} )
        # now evaluate it
        eval "${cmd[@]}"
    fi
done <<END
 - abc(i,j,k)
END
- abc(i,j,k)
- abc(i,j,)
- abc(i,,k)
- abc(i,,)
- abc(,j,k)
- abc(,j,)
- abc(,,k)
- abc(,,)
于 2013-07-20T18:32:42.147 回答
2

也许有一个更短的版本。我添加了纯解决方案:

#!/bin/bash

# extract arguments to $arg
fnc=" - abc(i,j,k)"
pre=${fnc%%(*}
post=${fnc##*)}
arg=${fnc#*(}
arg=${arg%)*}

# put arguments to arr array
IFS=, read -a arr <<<"$arg"
num=${#arr[@]}

# Generate all output lines
for((i=0; i<1<<num; ++i));{
  tmp=()
  for ((j=0; j<num; ++j));{
    if ((i & (1<<j))); then tmp[j]=""
    else tmp[j]=${arr[j]}
    fi
  }
  printf -v tmp ,%s "${tmp[@]}"
  echo "$pre("${tmp:1}")$post"
}

输出:

 - abc(i,j,k)
 - abc(,j,k)
 - abc(i,,k)
 - abc(,,k)
 - abc(i,j,)
 - abc(,j,)
 - abc(i,,)
 - abc(,,)

该脚本使用变量fnc作为输入。它分为三个部分:pre,args,post。args 是数组化的。然后它为每个可能的变化产生输出。

于 2013-07-20T14:43:28.787 回答