什么是shell script
shell script 是利用 shell 的功能所写的一个“程序”,这个程序是使用纯文本文件,将一些shell的语法与命令(含外部命令)写在里面,搭配正则表达式、管道命令、与数据流重定向等功能,以达到我们想要处理的目的。
Linux 的一个重要哲学思想就是一切模块化,使用一个个简单的模块来完成一个复杂的任务,这种思想在 shell 脚本中表现的淋漓精致,包括 Linux 中很多程序都是拥有一个个小模块,而这些模块你可以选择性的使用。
但是shell脚本也有自己的局限性,一些过于复杂和零碎的功能用shell脚本很难完成或者反而会很复杂,所有说没有最好的语言,只有最合适的语言。
shell script编写规范
shell 脚本开头需要“#!”加上脚本所使用的shell路径来声明解释此脚本的shell程序(因为主流的 Linux 发行版默认 shell 都是 bash shell,所以此处一般都是 “#!/bin/bash” )。
- 命令的执行时从上而下、从左而右的依次分析与执行。
- 命令、参数间的多个空格会被忽略。
- 空白行也将会被忽略。
- 如果读取到一个Enter符号(回车控制符),就尝试执行该命令串。
- 如果命令太长,则可使用反斜线“[Enter]”开扩展至下一行。
- “#”可作为批注,任何加在“#”后面的数据全部会被识别为注释文字而忽略。
在一些环境的配置上面,毕竟每个人的环境都不相同,为了取得较佳的运行环境, 我都会自行先定义好一些一定会被用到的环境变量,例如 PATH 这个玩意儿!
所以,建议你一定要养成良好的 script 撰写习惯,在每个 script 的文件头处记录好:
- script 的功能;
- script 的版本信息;
- script 的作者与联络方式;
- script 的版权宣告方式;
- script 的 History (历史纪录);
- script 内较特殊的命令,使用『绝对路径』的方式来下达;
- script 运行时需要的环境变量预先宣告与配置。
简单的shell script 范例
交互式shell脚本 使用read 命令,让脚本读取键盘输入。
[root@ZhangSir ~]#vim read.sh
#!/bin/bash
read -p "Please input your name:" name
echo -e "\nYour name is:$name"
[root@ZhangSir ~]#. read.sh
Please input your name:ZhangSir
Your name is:ZhangSir
输入两个数,显示相乘结果:
[root@ZhangSir ~/bin]#vim num.sh
#!/bin/bash
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH
echo -e "You SHOULD input 2 numbers, I will cross them! \n"
read -p "first number:" num1
read -p "second number:" num2
total=$(($num1*$num2))
echo -e "\nThe result of $num1 x $num2 is ==> $total"
[root@ZhangSir ~/bin]#. num.sh
You SHOULD input 2 numbers, I will cross them!
first number:9
second number:8
The result of 9 x 8 is ==> 72
脚本执行差异
使用bash或“.”加上路径:这种方式,脚本会在当前shell进程的子进程中进行。 source 加上脚本路径,这种方式,脚本实在当前shell中执行。
test 命令的测试功能
- 关于某个文件名的『文件类型』判断,如 test -e filename 表示存在否
测试的标志 | 代表意义 |
---|---|
-e | 该『文件名』是否存在?(常用) |
-f | 该『文件名』是否存在且为文件(file)?(常用) |
-d | 该『文件名』是否存在且为目录(directory)?(常用) |
-b | 该『文件名』是否存在且为一个 block device 装置? |
-c | 该『文件名』是否存在且为一个 character device 装置? |
-S | 该『文件名』是否存在且为一个 Socket 文件? |
-p | 该『文件名』是否存在且为一个 FIFO (pipe) 文件? |
-L | 该『文件名』是否存在且为一个连结文件? |
- 关于文件的权限检测,如 test -r filename 表示可读否 (但 root 权限常有例外)
测试的标志 | 代表意义 |
---|---|
-r | 检测该文件名是否存在且具有『可读』的权限? |
-w | 检测该文件名是否存在且具有『可写』的权限? |
-x | 检测该文件名是否存在且具有『可运行』的权限? |
-u | 检测该文件名是否存在且具有『SUID』的属性? |
-g | 检测该文件名是否存在且具有『SGID』的属性? |
-k | 检测该文件名是否存在且具有『Sticky bit』的属性? |
-s | 检测该文件名是否存在且为『非空白文件』? |
- 两个文件之间的比较,如: test file1 -nt file2
测试的标志 | 代表意义 |
---|---|
-nt | (newer than)判断 file1 是否比 file2 新 |
-ot | (older than)判断 file1 是否比 file2 旧 |
-ef | 判断 file1 与 file2 是否为同一文件,可用在判断 hard link 的判定上。 主要意义在判定,两个文件是否均指向同一个 inode 哩! |
- 关于两个整数之间的判定,例如 test n1 -eq n2
测试的标志 | 代表意义 |
---|---|
-eq | 两数值相等 (equal) |
-ne | 两数值不等 (not equal) |
-gt | n1 大于 n2 (greater than) |
-lt | n1 小于 n2 (less than) |
-ge | n1 大于等于 n2 (greater than or equal) |
-le | n1 小于等于 n2 (less than or equal) |
- 判定字串的数据
测试的标志 | 代表意义 |
---|---|
test -z string | 判定字串是否为 0 ?若 string 为空字串,则为 true |
test -n string | 判定字串是否非为 0 ?若 string 为空字串,则为 false。 注: -n 亦可省略 |
test str1 = str2 | 判定 str1 是否等于 str2 ,若相等,则回传 true |
test str1 != str2 | 判定 str1 是否不等于 str2 ,若相等,则回传 false |
- 多重条件判定,例如: test -r filename -a -x filename
测试的标志 | 代表意义 |
---|---|
-a | (and)两状况同时成立!例如 test -r file -a -x file,则 file 同时具有 r 与 x 权限时,才回传 true。 |
-o | (or)两状况任何一个成立!例如 test -r file -o -x file,则 file 具有 r 或 x 权限时,就可回传 true。 |
! | 反相状态,如 test ! -x file ,当 file 不具有 x 时,回传 true |
判断符号 [ ]
我们可以利用判断符号“[ ]”(中括号) 来进行数据的判断, 举例来说,如果我想要知道 $HOME 这个变量是否为空的,可以这样做:
root@ZhangSir ~]# [ -z "$HOME" ] ; echo $?
使用中括号必须要特别注意,因为中括号用在很多地方,包括万用字节与正规表示法等等,所以如果要在 bash 的语法当中使用中括号作为 shell 的判断式时,必须要注意中括号的两端需要有空白字节来分隔!
- 在中括号 [ ] 内的每个组件都需要有空白键来分隔;
- 在中括号 [ ] 内的变量,最好都以双引号括号起来;
- 在中括号 [ ] 内的常数,最好都以单或双引号括号起来。
Shell script 的默认变量
- $# :代表后接的参数『个数』,以上表为例这里显示为『 4 』;
- $@ :代表『 "$1" "$2" "$3" "$4" 』之意,每个变量是独立的(用双引号括起来);
- $* :代表『 "$1c$2c$3c$4" 』,其中 c 为分隔字节,默认为空白键, 所以本例中代表『 "$1 $2 $3 $4" 』之意。
- shift NUM:造成参数变量号码偏移,前面的变量会被取掉。
