bash shell 提供了一些不同的方法从用户处获取数据,这些方法包括命令行参数、命令行选项和直接读取键盘输入。
命令行参数
bash shell 将在命令行中输入的所有参数赋值给一些特殊变量,称为位置参数,通过标准数据表示,其中$0为程序名称,$1为第一个参数,$2为第二个参数,依此类推,直到$9为第九个参数。在第九个变量之后,必须使用大括号将变量括起来,如${10}。
示例
#!/bin/bash
# using one command line parameter
factorial=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 parameter
echo The command entered is : $0
调用:./test5
如果需要去除通过命令行运行脚本时使用的所有路径,使用 basename 命令。
示例
#!/bin/bash
# using basename with the $0 parameter
name=`basename $0`
echo The command entered is: $name
调用:./test5b
The command entered is : ./test11-5
参数计数变量
特殊变量 $# 中存储执行脚本时包含的命令行参数的个数。
示例
#!/bin/bash
# testing parameters
if [ $# -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 ]
done
count=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 parameters
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 1 ;;
-c) echo "Found the -c option" ;;
--) shift
break ;;
*) echo "$1 is not an option" ;;
esac
shift
done
count=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 getopt
set -- `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
done
count=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 getopts
while 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 command
echo -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 option
read -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 variable
read -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 entry
if 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 input
read -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 monitor
read -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 file
count=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