我有一个复杂的命令,我想做一个shell/bash脚本。我可以很容易地用1美元表示

foo $1 args -o $1.ext

我希望能够向脚本传递多个输入名称。正确的做法是什么?

当然,我想要处理包含空格的文件名。


当前回答

对参数数量变量$#进行循环也可以。

#! /bin/bash

for ((i=1; i<=$#; i++))
do
  printf "${!i}\n"
done
./test.sh 1 2 '3 4'

增长:

1
2
3 4

其他回答

对于简单的情况,你也可以使用shift。 它将参数列表视为队列。每次移位都会抛出第一个参数 其余每个参数的索引都递减。

#this prints all arguments
while test $# -gt 0
do
    echo "$1"
    shift
done

使用“$@”来表示所有的参数:

for var in "$@"
do
    echo "$var"
done

这将遍历每个参数,并将其打印在单独的行上。$@的行为类似于$*,除了当引用参数时,如果参数中有空格,则会被正确拆分:

sh test.sh 1 2 '3 4'
1
2
3 4
aparse() {
while [[ $# > 0 ]] ; do
  case "$1" in
    --arg1)
      varg1=${2}
      shift
      ;;
    --arg2)
      varg2=true
      ;;
  esac
  shift
done
}

aparse "$@"

请注意,Robert的答案是正确的,它也适用于sh。你可以(可移植地)进一步简化它:

for i in "$@"

等价于:

for i

也就是说,你什么都不需要!

测试($ is命令提示符):

$ set a b "spaces here" d
$ for i; do echo "$i"; done
a
b
spaces here
d
$ for i in "$@"; do echo "$i"; done
a
b
spaces here
d

我第一次读到这个是Kernighan和Pike写的Unix Programming Environment。

在bash中,帮助文档如下:

对于名字[用语言来说…]do COMMANDS;完成 如果'in WORDS…;'不存在,则假定'in "$@"'。

在脚本中使用命令格式化任何命令行选项或 参数。

#!/bin/bash
# Extract command line options & values with getopt
#
set -- $(getopt -q ab:cd "$@")
#
echo
while [ -n "$1" ]
do
case "$1" in
-a) echo "Found the -a option" ;;
-b) param="$2"
echo "Found the -b option, with parameter value $param"
shift ;;
-c) echo "Found the -c option" ;;
--) shift
break ;;
*) echo "$1 is not an option";;
esac
shift