@yangfch3
2017-01-18T16:23:49.000000Z
字数 5042
阅读 3392
shell
接前文:Shell 编程 - 01
01 篇介绍的是基础知识:
本章介绍其他语法与流程结构
echo 用于在屏幕上打印出指定的字符串。
最好的理解 echo
的方法就是:
mouth=8
echo "${mouth}-1-2009"
echo "OK!\n"
echo "It is a test" > myfile
echo '$name\"' # 注意单引号
echo `date`
在 ""
内可以使用 $()
(推荐) 或 `` 包裹命令
echo "There are $(ls | wc -l) items here."
printf 命令用于格式化输出,是 echo 命令的增强版。它是 C 语言 printf() 库函数的一个有限的变形
# format-string为双引号
$ printf "%d %s\n" 1 "abc"
1 abc
# 单引号与双引号效果一样
$ printf '%d %s\n' 1 "abc"
1 abc
# 没有引号也可以输出
$ printf %s abcdef
abcdef
# 格式只指定了一个参数,但多出的参数仍然会按照该格式输出,format-string 被重用
$ printf %s abc def
abcdef
$ printf "%s\n" abc def
abc
def
$ printf "%s %s %s\n" a b c d e f g h i j
a b c
d e f
g h i
j
# 如果没有 arguments,那么 %s 用NULL代替,%d 用 0 代替
$ printf "%s and %d \n"
and 0
# 如果以 %d 的格式来显示字符串,那么会有警告,提示无效的数字,此时默认置为 0
$ printf "The first program always prints'%s,%d\n'" Hello Shell
-bash: printf: Shell: invalid number
The first program always prints 'Hello,0'
三种 if
:
if ... else 语句的语法:
if [ expression ]
then
Statement(s) to be executed if expression is true
fi
if ... else ... fi 语句的语法:
if [ expression ]
then
Statement(s) to be executed if expression is true
else
Statement(s) to be executed if expression is not true
fi
if ... elif ... fi 语句可以对多个条件进行判断,语法为:
if [ expression 1 ]
then
Statement(s) to be executed if expression 1 is true
elif [ expression 2 ]
then
Statement(s) to be executed if expression 2 is true
elif [ expression 3 ]
then
Statement(s) to be executed if expression 3 is true
else
Statement(s) to be executed if no expression is true
fi
单行书写,使用 ;
分隔
if test $[2*3] -eq $[1+5]; then echo 'The two numbers are equal!'; fi;
凡是使用分行的情况都可以使用
;
连为一行
if ... else 语句也经常与 test
命令结合使用,如下所示:
num1=$[2*3]
num2=$[1+5]
if test $[num1] -eq $[num2]
then
echo 'The two numbers are equal!'
else
echo 'The two numbers are not equal!'
fi
case...esac 用于简写分支(不需要那么多 if)
case 值 in
模式1)
command1
command2
command3
;;
模式2)
command1
command2
command3
;;
*)
command1
command2
command3
;;
esac
option="${1}"
case ${option} in
-f) FILE="${2}"
echo "File name is $FILE"
;;
-d) DIR="${2}"
echo "Dir name is $DIR"
;;
*)
echo "`basename ${0}`:usage: [-f file] | [-d directory]"
exit 1 # Command to come out of the program with status 1
;;
esac
运行输出:
$./test.sh
test.sh: usage: [ -f filename ] | [ -d directory ]
$ ./test.sh -f index.htm
$ vi test.sh
$ ./test.sh -f index.htm
File name is index.htm
$ ./test.sh -d unix
Dir name is unix
遍历参数序列
for 变量 in 参数序列
do
command1
command2
...
commandN
done
其中参数序列的写法:
{m..n}
:表示从 m 到 n 的数值区间x y z
:表示 x y z 三个值做为遍历时的参数区间循环
for ((a=1; a <= 3; a++))
do
command1
command2
done
例如:
for loop in 1 2 3 4 5
do
echo "The value is: $loop"
done
for str in 'This is a string'
do
echo $str
done
for FILE in $HOME/.bash*
do
echo $FILE
done
语法格式:
while [command] # 这里的 command 与 if 哪里的条件语句一样
do
Statement(s) to be executed if command is true
done
例如:条件递增
COUNTER=0
while [ $COUNTER -lt 5 ]
do
COUNTER='expr $COUNTER+1'
echo $COUNTER
done
例如:读取键盘信息。下面的例子中,输入信息被设置为变量FILM,按 Ctrl-D 结束循环
echo 'type <CTRL-D> to terminate'
echo -n 'enter your most liked film: '
while read FILM
do
echo "Yeah! great film the $FILM"
done
until 循环执行一系列命令直至条件为 true 时停止。until 循环与 while 循环在处理方式上刚好相反。
until command
do
Statement(s) to be executed until command is true
done
用在循环内,break
, continue
用于跳出循环。break
是完全的跳出并结束循环,continue
是跳出当前,后面还没完成的循环还会继续。
不像 JS 中的 lable,break
后面可以接数值,用于表示调出循环的层级。如果没有接数值则表示跳出当前层循环。continue
后同样可以接数值。
for var1 in 1 2 3
do
for var2 in 0 5
do
if [ $var1 -eq 2 -a $var2 -eq 0 ]
then
break 2
else
echo "$var1 $var2"
fi
done
done
while :
do
echo -n "Input a number between 1 to 5: "
read aNum
case $aNum in
1|2|3|4|5) echo "Your number is $aNum!"
;;
*) echo "You do not select a number between 1 to 5!"
continue
echo "Game is over!"
;;
esac
done
其实 bash 许多命令的本质就是一个函数,利用函数我们能拆解一个复杂功能为小模块,达到原子代码的复用。
# 形式1:直接函数名,无关键字
function_name () {
list of commands
[ return value ]
}
# 形式2:function 关键字
function function_name () {
list of commands
[ return value ]
}
Shell 函数必须先定义后使用。
函数的返回值:可以显式增加 return 语句;如果不加,会将最后一条命令运行结果作为返回值。
Shell 函数返回值只能是整数,一般用来表示函数执行成功与否,0表示成功,其他值表示失败。
如果 return 其他数据,比如一个字符串,往往会得到错误提示:“numeric argument required”
。
只需要给出函数名,不需要加括号。在调用函数时可以在函数名后面添加任意数量的参数,以空格间隔
function_name
function_name var1 var2 var3
number_one () {
echo "Url_1 is http://see.xidian.edu.cn/cpp/shell/"
number_two
}
number_two () {
echo "Url_2 is http://see.xidian.edu.cn/cpp/u/xitong/"
}
number_one
unset .f function_name
函数调用时的传参:
funWithParam 1 2 3 4 5 6 7 8 9 34 73
函数内部使用参数:$(n)
与特殊变量
funWithParam(){
echo "The value of the first parameter is ${1} !"
echo "The value of the second parameter is ${2} !"
echo "The value of the tenth parameter is $10 !" # 观察 $10 与 $(10) 的输出区别
echo "The value of the tenth parameter is ${10} !"
echo "The value of the eleventh parameter is ${11} !"
echo "The amount of the parameters is $# !" # 参数个数
echo "The string of the parameters is $* !" # 传递给函数的所有参数
}
特殊变量:
>
:暴力重定向>>
:追加输出到文件尾命令的输出不仅可以是显示器(标准输出回显),还可以很容易的转移向到文件,这被称为输出重定向。
which zsh > location.txt
which bash >> location.txt
<
:和输出重定向一样,Unix 命令也可以从文件获取输入。
echo < location.txt
(n, m 可以为 0, 1, 2,分别表示标准输入,标准输出,标准错误输出)
(这里的标准输入文件,标准输出文件,标准错误文件即上面的 0, 1, 2)
重定向高级一点的写法:
# 希望将 stdout 和 stderr 合并后重定向到 file
command > file 2>&1
# 对 stdin 和 stdout 都重定向
command < file1 >file2
<<
delimiter
:自定义的标志,常见 EOF
command << delimiter
document
delimiter
它的作用是将两个 delimiter 之间的内容(document) 作为输入传递给 command。
例如:
$wc -l << EOF
This is a simple lookup program
for good (and bad) restaurants
in Cape Town.
EOF
3
Shell 也可以包含外部脚本,将外部脚本的内容在当前脚本执行
#### subscript.sh
url="http://see.xidian.edu.cn/cpp/view/2738.html"
#### main.sh
. ./subscript.sh # subscript.sh 表示的是另外一个 sh
echo $url