bash shell 提供了一些不同的方法从用户处获取数据,这些方法包括命令行参数、命令行选项和直接读取键盘输入。

 

命令行参数

bash shell 将在命令行中输入的所有参数赋值给一些特殊变量,称为位置参数,通过标准数据表示,其中$0为程序名称,$1为第一个参数,$2为第二个参数,依此类推,直到$9为第九个参数。在第九个变量之后,必须使用大括号将变量括起来,如${10}。

示例

#!/bin/bash
# using one command line parameterfactorial=1
for (( number=1; number<=$1; number++ ))
do
  factorial=$[ factorial * $number ]
done
echo The factorial of $1 is $factorial

调用:$ ./test1 5

The factorial of 5 is 120

 

读取程序名称

使用参数$0可以确定 shell 从命令行启动的程序的名称。

示例

#!/bin/bash
# testing the $0 parameterecho The command entered is : $0

调用:./test5

 

如果需要去除通过命令行运行脚本时使用的所有路径,使用 basename 命令。

示例

#!/bin/bash
# using basename with the $0 parametername=`basename $0`
echo The command entered is: $name

调用:./test5b

The command entered is : ./test11-5

 

参数计数变量

特殊变量 $# 中存储执行脚本时包含的命令行参数的个数。

示例

#!/bin/bash
# testing parametersif [ $# -ne 2 ]
then
  echo Uage: test11-9 a b
else
  total=$[ $1 + $2 ]
  echo The total is $total
fi

另外,使用变量 ${!#} 可以得到最后一个命令行参数值。注意不能使用 ${$#} 这种格式,否则会得到错误的结果。

 

调用:./test9 ×
调用:./test9 10 ×
调用:./test9 10 15 20  ×
Uage: test11-9 a b
调用:./test9 10 15  √
The total is 25

 

获取所有数据变量

变量 $* 将命令行中提供的所有参数作为一个单词处理。这个单词中包含出现在命令行中的每一个参数值。

变量 $@ 将命令行中提供的所有参数作为同一个字符串中的多个单词处理。允许对其中的值进行迭代,分隔开所提供的不同参数。

示例

#!/bin/bash
# testing $* and $@count=1
for param in "$*"
do
  echo "\$* Parameter #$count= $param"
  count=$[ $count + 1 ]
donecount=1
for param in "$@"
do
  echo "\$@ Parameter #$count=$param"
  count=$[ $count + 1 ]
done

 

调用:./test12 rich barbara katie jessica

$*  Parameter #1=rich barbara katie jessica
$@ Parameter #1=rich
$@ Parameter #2=barbara
$@ Parameter #3=katie
$@ Parameter #4=jessica

 

命令行选项

选项是由破折号引导的单个字母,它更改命令的行为。

执行 shell 脚本时经常会遇到既需要使用选项又需要使用参数的情况。在 Linux 中的标准方式是使用特殊符号双破折号(--)将二者分开,这个特殊字符号码告诉脚本选项结束和普通参数开始的位置。

示例

#!/bin/bash
# extracting options and parameterswhile [ -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 1 ;;
  -c) echo "Found the -c option" ;;
  --) shift
    break ;;
  *) echo "$1 is not an option" ;;
  esac
  shift
donecount=1
for param in $@
do 
  echo "Parameter #$count: $param"
  count=$[ $count + 1 ]
done

 

调用:./test16 -c -a -b test1 test2 test3

Found the -c option
Found the -a option
Found the -b option, with parameter value test1
test2 is not an option
test3 is not an option

调用:./test16 -c -a -b -- test1 test2 test3

Found the -c option
Found the -a option
Found the -b option, with parameter value --
test1 is not an option
test2 is not an option
test3 is not an option

使用 getopt 命令

getopt 命令可以接受任意形式的命令行选项和参数列表,并自动将这些选项和参数转换为适当的格式。

格式:

  getopt options optstring parameters

optstring :选项字符串,定义命令行中的有效选项字母。在每个需要参数值的选项字母后面放置一个冒号,如:

  getopt ab:cd -a -b test1 -cd -test2 -test3

当执行 getopt 命令时,会监测提供的参数列表,然后基于提供的选项字符串对列表进行解析。

  -a -b test1 -c -d test2 test3

如果指定的选项不在选项字符串中,默认会生成一个错误消息。

getopt ab:cd -a -b test1 -cde test2 test3
getopt: invalid option -- e
-a -b test1 -c -d test2 test3

 

在脚本中使用 getopt

将原始脚本命令行参数送给 getopt 命令,然后将 getopt 命令的输出送给 set 命令:

 set -- `getopt -q ab:cd "$@"`

set 命令的一个选项是双破折号,表示将命令行参数变量替换为 set 命令的命令行中的值。

示例

#!/bin/bash
# extracting command line options and values with getoptset -- `getopt -q ab:c "$@"`
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
donecount=1
for param in "$@"
do
 echo "Parameter #$count: $param"
 count=$[ $count + 1 ]
done

调用:./test18 -ac

Found the -a option
Found the -c option

调用:./test18 -a -b test1 -cd test2 test3 test4

Found the -a option
Found the -b option, with parameter value 'test1'
Found the -c option
Parameter #1: 'test2'
Parameter #2: 'test3'
Parameter #3: 'test4'

 

高级的 getopts 命令

getopts 命令顺序的对现有的 shell 参数变量进行处理。非常适宜用在循环中解析所有命令行参数。

getopts 命令的格式为:

   getopts optstring variable

optstring 与 getopt 中的相似。如果要禁止输出错误消息,那么使选项字符串以冒号开头。

getopts 使用两个环境变量:

  OPTARG:包含需要参数值的选项要使用的值

  OPTIND:包含的值表示 getopts 停止处理时在参数列表中的位置

示例

#!/bin/bash
# processing options and parameters with getoptswhile getopts :ab:cd opt
do
 case "$opt" in
 a) echo "Found the -a option" ;;
 b) echo "Found the -b option, with value $OPTARG" ;;
 c) echo "Found the -c option" ;;
 d) echo "Found the -d option" ;;
 *) echo "Unknown option: $opt" ;;
 esac
done
shift $[ $OPTIND - 1 ]count=1
for param in "$@"
do
 echo "Parameter $count: $param"
 count=$[ $count + 1 ]
done

调用:./test20 -a -b test1 -d test2 test3 test4

Found the -a option
Found the -b option, with value test1
Found the -d option
Parameter 1: test2
Parameter 2: test3
Parameter 3: test4

getopt 与 getopts 区别

getopt是个外部binary文件,而getopts是built-in。getopts 有两个参数,第一个参数是一个字符串,包括字符和“:”,每一个字符都是一个有效的选项,如果字符后面带有“:”,表示这个字符有自己的参数。 getopts从命令中获取这些参数,并且删去了“-”,并将其赋值在第二个参数中,如果带有自己参数,这个参数赋值在“OPTARG”中。提供getopts的shell内置了OPTARG这个变量,getopts修改了这个变量。而getopt则不能。因此getopt必须使用set来重新设定位置参数$1,$2....,然后在getopt中用shift的方式依次来获取。基本上,如果参数中可能含有空格,那么必须用getopts。否则仅仅从使用上看,他们没有区别。

键盘输入

 read 命令接受标准输入(键盘)的输入,或其他文件描述符的输入。

示例

#!/bin/bash
# testing the read commandecho -n "Enter your name: "
read name
echo "Hello $name, welcome to my program."

 

调用:./test21

Enter your name: vian
Hello vian, welcome to my program.

 

在 read 命令包含 -p 选项,允许在 read 命令行中直接指定一个提示

示例

#!/bin/bash
# testing the read -p optionread -p "Please enter your age: " age
days=$[ $age * 365 ]
echo "That makes you over $days days old!"

 

调用:./test22

Please enter your age: 18
That makes you over 6570 days old!

 

在 read 命令行中也可以不指定变量。如果不指定变量,那么 read 命令会将接收到的数据放置在环境变量 REPLY 中

示例

#!/bin/bash
# testing the REPLY environment variableread -p "Enter a number:"
factorial=1
for (( count=1; count<= $REPLY; count++ ))
do 
 factorial=$[ $factorial * $count ]
done
echo "The factorial of $REPLY is $factorial"

调用:./test24

Enter a number:12
The factorial of 12 is 479001600

计时功能

使用 -t 选项指定 一个计时器。

示例

#!/bin/bash
# timing the data entryif read -t 5 -p "Please enter your name: " name
then
 echo "Hello $name, welcome to my script"
else
 #echo
 echo "Sorry, too slow!"
fi

 

调用:./test25

Please enter your name: ivan
Hello ivan, welcome to my script

除了输入时间计时,还可以设置 read 命令计数输入的字符。

示例

#!/bin/bash
# getting just one character of inputread -n1 -p "Do you want to continue [Y/N]? " answer
case $answer in
Y | y) echo
 echo "fine, continue on..." ;;
N | n) echo
 echo "OK, goodbye" 
 exit;;
esac
echo "This is the end of the script"

 

调用:./test26

Do you want to continue [Y/N]? y
fine, continue on...
This is the end of the script

默读

-s 选项能够使输入数据不显示在屏幕上。

示例

#!/bin/bash
# hiding input data from the monitorread -s -p "Enter your password: " pass
echo
echo "Is your password really $pass?"

 

调用:./test27

Enter your password: 
Is your password really abc?

读取文件

读取文件的关键是如何将文件中的数据传送给 read 命令。

示例

#!/bin/bash
# reading data from a filecount=1
cat states | while read line
do
 echo "Line $count: $line"
 count=$[ $count + 1 ]
done
echo "Finished processing the file"

 

调用:./test28

Line 1: Alabama
Line 2: Alaska
Line 3: Arizona
Line 4: Arkansas
Line 5: Colorado
Line 6: Connecticut
Line 7: Florida
Line 8: Georgia
Finished processing the file