每一条命令;不管是内置的,shell 函数,还是外部的,当它退出时,都会返回一个小的整数值给引用它的程序,这就是大家所熟知的程序的退出状态.在 shell 下执行进程是,有很多方式可取用程序的退出状态.
以管理来说,退出状态为 0 表示”成功”,也就是,程序执行完成且为遭遇到任何问题.其他任何的退出退出状态都为失败.内置变量?(使用命令echo $?)查看上一条命令的退出状态.
案例:当你输入ls -l /dev/null
时.
输出:crw-rw-rw- 1 root root 1, 3 6月 25 15:41 /dev/null
接着使用命令:echo $?
输出为 0
接着使用命令:ls foo
输出:ls: 无法访问 foo: 没有那个文件或目录
echo $?
输出:2
表示没有成功的执行.
POSIX 的结束状态
值 |
意义 |
0 |
命令成功地退出 |
>0 |
在重定向或单词展开期间(~,变量,命令,算数展开,以及单词切割)失败 |
1-125 |
命令不成功的退出.特定的退出值的含义,是由各个单独的命令定义的. |
126 |
命令找到了,但文件无法执行 |
127 |
命令没找到 |
>128 |
命令因受到信号而死亡 |
POSIX 留下退出状态 128 未定义,仅要求他表示某种失败.因为只有低位的 8 个位会返回给父进程,所以大于255的退出状态都会替换成该值除以 256 之后的余数.
在 shell 脚本可以使用 exit 命令传递一个退出之给踏的调用者.只要将一个数字传递给它,作为一个参数即可.脚本会立即退出,并且调用者会受到该数字且作为脚本的退出值.
说白了 exit 就是退出当前的 shell,在 shell 脚本中可以终止当前脚本执行.
exit
语法:
exit [exit-value]
用途:
目的是从 shell 脚本返回一个退出状态给脚本的调用者.
主要选项:
无
行为模式:
如果没有提供,则以最后一个执行命令的退出状态作为默认的退出状态.如果这就是你要的,则最好明白的在 shell 脚本里这么写:exit $?
案例:exit
输出为 logout,表示退出当前 shell
案例二:脚本代码cd $(dirname $0) || exit 1
进入脚本所在目录,否则退出
案例三:脚本中判断参数数量,不匹配就打印使用方式,退出
代码:
if [ "$#" -ne "2" ]; then
echo "usage: $0 <area> <hours>"
exit 2
fi
案例四:在脚本里,退出时删除临时文件
代码:trap “入门-rf tempfile;echo Bye.” exit
案例五:检查上一行的退出码
代码:
EXCODE=$?
if [ "$EXCODE" == "0" ]; then
echo "O.K"
fi
if-elif-else-fi 语句
if 语法:
if pipeline
[pipeline...]
then
statement-if-true-1
elif pipeline
[pipeline...]
then
statement-iftrue2
else
statement-if-all-else-fails
if
使用方括号作为开始与结束的关键字将语句组织起来.
案例 1:
提示用户指定备份目录的路径,若目录存在则显示信息跳过,否则显示相应提示信息,并创建该目录.
bash 代码:
#!/bin/bash
read -p "what is your backup directoy : " BakDir
if [ -d $BakDir ];then
echo "$BakDir alerdy exist"
else
echo "$BakDir is not exist,will make it"
mkdir $BakDir
fi
案例 2:统计当前登录到系统中的用户数量,若判断是否超过三个,若是则显示实际数量并给出警告信息,否则列出 登录的用户账户名称及所在终端 bash 代码:
UserNum='who | wc -l'
if [ $UserNum -gt 3 ];
then
echo "Alert, too many login users ( Total: $UserNum)."
else
echo "Login Users:"
who | awk '{print $1,$2}'
fi
注意:
1、if 与[ 之间必须有空格
2、[ ]与判断条件之间也必须有空格
3、]与; 之间不能有空格
逻辑的 not,and 与 or
“如果 john 不在家,则...” ,在 shell 下这种情况的做法是:将惊叹号放在管道前:
if ! grep pattern myfile > /dev/null
then
模式不在这里
fi
“如果 john 在家,且他不忙,则....”,使用逻辑 and.
if grep pattern1 myfile && grep pattern2 myfile
then
myfile 包含两种模式
fi
相对的,||运算符则用来测试两个条件中是否有一个为真.:
if grep pattern1 myfile || grep pattern2 myfile
then
myfile 包含两种模式之一
fi
逻辑 and 和 or 都是快捷运算符,即当判断出整个语句块的真伪时,shell 会立即停止执行命令.举例来说,在 command1&&command2 下,如果 aommand1 失败,则整个结果不可能为真,所以 command2 也不会被执行;以此类推,command1||command2 指的是:如果 command1 成功,那么也没有理由执行 command2.
不要尝试过度”简练”未使用&&和||来取代if语句.我们不反对简短且简单的事情,例如:
命令:who | grep root > /dev/null && echo root is login on root is login on
输出:root is login on
分析:上面的命令实际做法是:执行 who | grep...且如果成功,就显示信息.而我们曾见过厂商提供 shell 脚本,所使用的是这样的结构:
```some_command &&{
one command
a decond command
and a third command
}
这个命令的意思是说将所有的语句块放在一块,只有在 some_command 成功时他们才被执行.使用if可以让他更简洁:
if some_command
then
one command
a second command
and a third command
fi
最后在判断语句中常用的运算符:
1、字符串判断
str1 = str2 当两个串有相同内容、长度时为真
str1 != str2 当串 str1 和 str2 不等时为真
-n str1 当串的长度大于 0 时为真(串非空)
-z str1 当串的长度为 0 时为真(空串)
str1 当串 str1 为非空时为真
2、数字的判断
int1 -eq int2 两数相等为真
int1 -ne int2 两数不等为真
int1 -gt int2int1 大于 int2 为真
int1 -ge int2int1 大于等于 int2 为真
int1 -lt int2int1 小于 int2 为真
int1 -le int2int1 小于等于 int2 为真
3 文件的判断
-r file 用户可读为真
-w file 用户可写为真
-x file 用户可执行为真
-f file 文件为正规文件为真
-d file 文件为目录为真
-c file 文件为字符特殊文件为真
-b file 文件为块特殊文件为真
-s file 文件大小非 0 时为真
-t file 当文件描述符(默认为 1 )指定的设备为终端时为真
3、复杂逻辑判断
-a 与
-o 或
! 非
test 命令
test 命令可以处理 shell 脚本里的各类工作.它产生的不是一般输出,而是可使用的退出状态.test 接受各种不同的参数,可控制它要执行哪一种测试.
test 命令有另一种形式:[...],这种永福的作用完全与 test 命令一样.因此,下面这两个案例表达的意思相同
if test “$str1”=”$str2”
then
...
fi
和
if [ “$str1” = “$str2” ]
then
...
fi
一样
test 的语法:
```test [expression]```
```[ [expression] ]```
用途:
为了测试 shell 脚本里的条件,通过退出状态返回其结果.要特别注意的是:这个命令的第二种形式,方括号根据字面意义逐字的输入,且必须与括起来的 expression 以空白隔开.
主要选项:
和使用用于if的选项一致.
其中
<table>
<tbody>
<tr>
<td valign="top">
<p>选项</p>
</td>
<td valign="top">
<p>含义</p>
</td>
</tr>
<tr>
<td valign="top">
<p>string </p>
</td>
<td valign="top">
<p>如果<span style="font-family:Times New Roman">...</span><span style="font-family:宋体">则为真</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-b file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">是块设备文件</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-d file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">是目录</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-c file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">是字符设备文件</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-e file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">存在</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-f file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">为一般文件</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-g file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">有设置他的</span><span style="font-family:Times New Roman">setgid</span><span style="font-family:宋体">位</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-h file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">是一符号链接</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-L file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">是一符号链接</span><span style="font-family:Times New Roman">(</span><span style="font-family:宋体">等同于 </span><span style="font-family:Times New Roman">-h)</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-n string</p>
</td>
<td valign="top">
<p>string<span style="font-family:宋体">是非</span><span style="font-family:Times New Roman">null</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-p file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">是一命名的管道</span><span style="font-family:Times New Roman">(FIGO</span><span style="font-family:宋体">文件</span><span style="font-family:Times New Roman">)</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-r file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">是可读的</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-S file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">是</span><span style="font-family:Times New Roman">socket</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-s file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">不是空的</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-t n</p>
</td>
<td valign="top">
<p>文件描述符<span style="font-family:Times New Roman">n</span><span style="font-family:宋体">指向一终端</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-u file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">有设置它的</span><span style="font-family:Times New Roman">setuid</span><span style="font-family:宋体">位</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-w file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">是可写入的</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-x file</p>
</td>
<td valign="top">
<p>file<span style="font-family:宋体">是可执行的</span><span style="font-family:Times New Roman">,</span><span style="font-family:宋体">或</span><span style="font-family:Times New Roman">file</span><span style="font-family:宋体">是可被查找的目录</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>-z string</p>
</td>
<td valign="top">
<p>string<span style="font-family:宋体">为</span><span style="font-family:Times New Roman">null</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>s1=s2 <span style="font-family:宋体">或者</span><span style="font-family:Times New Roman">s1!=s2</span></p>
</td>
<td valign="top">
<p>字符串相不相等</p>
</td>
</tr>
<tr>
<td valign="top">
<p>n1 -eq n2</p>
</td>
<td valign="top">
<p>整数<span style="font-family:Times New Roman">n1</span><span style="font-family:宋体">等于</span><span style="font-family:Times New Roman">n2</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>n1 -ne n2</p>
</td>
<td valign="top">
<p>整数<span style="font-family:Times New Roman">n1</span><span style="font-family:宋体">不等于</span><span style="font-family:Times New Roman">n2</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>n1 -lt n2 </p>
</td>
<td valign="top">
<p>n1<span style="font-family:宋体">小于</span><span style="font-family:Times New Roman">n2</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>n1 -gt n2</p>
</td>
<td valign="top">
<p>n1<span style="font-family:宋体">大于</span><span style="font-family:Times New Roman">n2</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>n1 -le n2</p>
</td>
<td valign="top">
<p>n1<span style="font-family:宋体">小于或等于</span><span style="font-family:Times New Roman">n2</span></p>
</td>
</tr>
<tr>
<td valign="top">
<p>n1 -ge n2</p>
</td>
<td valign="top">
<p>n1<span style="font-family:宋体">大于或等于</span><span style="font-family:Times New Roman">n2</span></p>
</td>
</tr>
</tbody>
</table>
案例:
bash 代码;
cd /bin
if test -e ./bash //其实这里相当于if [ -e ./bahs ]
then
echo 'the file already exist!'
else
echo 'the file not exist!'
fi
输出结果为:the file already exist!
另外,shell 还提供了-a(逻辑 AND ),-o(逻辑 OR),-a 的优先级高于-o,而=与!=优先级则高于其他的二元运算符.
注意:在使用-a 和-o(这两个事 test 运算符)与&&和||(这两个事shell运算符)之间有一个差异:
```if [ -n “$str” -a -f “$file” ]``` 一个test命令,两种条件
```if [-n “str”] && [ -f “$file” ]``` 两个命令,一块接方式计算
```if [-n “$str” && -f ”$file”]``` 语法错误
第一个案例,test 会计算两种条件.而第二个案例,shell 执行第一个 test 命令,且只有在第一个命令是成功的情况下,才会执行第二个命令.最后一个案例,&&为 shell 运算符,所以它会终止第一个 test 命令,然后这个命令会抱怨它找不到结束的]字符,且以失败的值退出.即使 test 可以成功的退出,接下来的检查还会失败,因为 shell(最有可能)找不到一个名为-f 的命令
## 精简表达式:
使用命令:```[1 eq1 ] &&echo’OK’```
输出:ok
使用命令:```[ 2 < 1 ] &&echo ‘OK’```
输出:-bash: 1: No such file or directory
使用命令:```[ 2 \< 1 ] &&echo ‘OK’```这样就可以了
使用命令:```[ 2 -gt 1 -a 3 -lt 4 ]&&echo 'Ok'```
输出:Ok
使用命令:```[ 2 -gt 1 && 3 -lt 4 ]&&echo 'Ok'```
输出:-bash: [: missing `]'
注意:在[] 表达式中,常见的>,<需要加转义字符,表示字符串大小比较,以 acill 码位置作为比较。不直接支持<>运算符,还有逻辑运算符 || 和 && 它需要用-a[and] –o[or]表示。
刚才使用的[],现在再来看使用[[]]
案例:
使用命令:[[ 2 < 3 ]]&&echo ‘OK’
输出 OK.
使用命令:[[ 2 < 3 && 4 < 5 ]] && echo 'ok'
输出:ok
注意:[[]] 运算符只是[]运算符的扩充。能够支持<,>符号运算不需要转义符,它还是以字符串比较大小。里面支持逻辑运算符 || 和 &&
bash 的条件表达式中有三个几乎等效的符号和命令:test,[]和[[]]。通常,大家习惯用 if [];then 这样的形式。而[[]]的出现,根据 ABS 所说,是为了兼容><之类的运算符。
不考虑对低版本 bash 和对 sh 的兼容的情况下,用[[]]是兼容性强,而且性能比较快,在做条件运算时候,可以使用该运算符。