shell条件测试

2023-12-13 03:30:46

目录

1.1.用途

1.2.基本语法

1.2.1.格式:

1.2.2.例

1.3 文件测试

1.4.整数测试

1.4.1.作用

1.4.2.操作符

1.4.3.示例:

1.5.逻辑操作符

1.5.1.符号

1.5.2.例:

1.6.命令分隔符


1.1.用途

????????为了能够正确处理Shell程序运行过程中遇到的各种情况,Linux Shell提供了一组测试运算符.
????????通过这些运算符,shell程序能够判断某种或者几个条件是否成立。
????????条件测试在各种流程控制语句,例如判断语句和循环语句中发挥了重要的作用,所以了解和掌握这些条件测试是非常重要的

1.2.基本语法

1.2.1.格式:

????????格式1:? ? test???????? -参数????????条件表达式
????????格式2:? ??[? ? 条件表达式? ? ]? ? # 注意: [? ?] 的左右要有空格
????????格式3:? ? [[? ? 条件表达式? ? ]]? # 注意: [? ?] 的左右要有空格,格式1.2的增强版本,支持通配符? ? ? ? ? ? ? ? #? 只有[[ ]]可以支持并且启用双重正则
????????格式4:????????((条件表达式))

注意:

????????test单独使用,判断条件为真,echo s?返回0,假返回1
????????test与[? ?]等价
????????[[? ?]] 是扩展命令,可以使用通配符等进行模式匹配,&& >等操作符可以直接应用于双中括号中,但不能用于单中括号中
????????(())一般用于if语句里,两端不需要有空格,测试对象为整数

????????在[? ?]中参数:-a并且,-o或者

1.2.2.例

格式一:

test???????? -参数????????条件表达式

测试passwd是否为普通文

[root@server ~]# test -f /etc/passwd? ? ? ? ? ? ? ? # -f 是否为普通文件
[root@server ~]# echo $?? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?#?以上俩条命令要配合使用

? ? ? ? ? ? ? ? ? ? ? ? # 返回值0成立,返回值1不成立。

# test配合自编逻辑表示

[root@server ~]# test -f /etc/hosts && echo 1 || echo 0
1
[root@server ~]# test -f /etc/a1.txt && echo 1 || echo 0
0
# 解析:?

? ? ? ? 并且(&&):左侧成立则右侧执行

? ? ? ? 或者(||):左侧成立则右侧不执行

扩展:

[root@server ~]# test -f /etc/hosts && echo "文件为普通文件" || echo "文件不为普通文件"
文件为普通文件
[root@server ~]# test -f /etc/a1.txt && echo "文件为普通文件" || echo "文件不为普通文件"
文件不为普通文件

格式二:

??[? ? 条件表达式? ? ]? ? # 注意: [? ?] 的左右要有空格? ? ? ? #?test与[ ]等价

[root@server ~]# [ -f /etc/hosts ] ?&& echo 1 || echo 0?
1
[root@server ~]# [ -f /etc/a1.txt ] ?&& echo 1 || echo 0?
0
[root@server ~]# [ -f /etc/hosts ] && echo "文件为普通文件" || echo "文件不为普通文件"?
文件为普通文件
[root@server ~]# [ -f /etc/a1.txt ] && echo "文件为普通文件" || echo "文件不为普通文件"?
文件不为普通文件
[root@server ~]# [ 3 > 2 && 1 < ?2 ] && echo yes ||echo no?
-bash: [: 缺少 `]'? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? # 不支持通配符等进行模式匹配
no

?格式三:? ?

[[? ? 条件表达式? ? ]]? ????????????????# 注意: [? ?] 的左右要有空格,格式1.2的增强版本,支持通配符

[root@server ~]# [[ 3>2 || 1>2 ]]
-bash: 条件命令中有未预期的符号 284
-bash: “3>”附近有语法错误????????????????
# 报错原因<, > 俩侧需要加空格

&&(并且):
[root@server ~]# [[ 3 > 2 && 1 > 2 ]] && echo yes ||echo no
no
[root@server ~]# [[ 3 < ?2 && 1 > 2 ]] && echo yes || echo no
no
[root@server ~]# [[ 3 < 2 && 1 > 2 ]] && echo yes || echo no
no
[root@server ~]# [[ 3 < 2 && ?1 < ?2 ]] && echo yes || echo no
no
[root@server ~]# [[ 3 > 2 && 1 < ?2 ]] && echo yes ||echo no
yes? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?# 结论:左右都成立才返回真

||(或者):

[root@server ~]# [[ 3 > 2 || 1 > 2 ]] && echo yes ||echo no
yes

[root@server ~]# [[ 3 > 2 || 1 > 2 ]] && echo yes || echo no
yes
[root@server ~]# [[ 3 < 2 || 1 < ?2 ]] && echo yes || echo no
yes
[root@server ~]# [[ 3 < 2 || 1 > 2 ]] && echo yes || echo no
no? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?# 结论:只要左右有一方成立则返回为真? ? ?? ? ? ??

[root@server ~]# [[ -f /etc/passwd ]] ?&& echo 1 || echo 0
1

?格式四:

????????((条件表达式))? ?# 一般只支持整数的操作

[root@server ~]# ((3>5)) ?&& echo 1 || echo 0
0
[root@server ~]# ((3<5)) ?&& echo 1 || echo 0
1

不可执行:
[root@server ~]# ((3.1<5)) ?&& echo 1 || echo 0
-bash: ((: 3.1<5:语法错误: 无效的算术运算符 (错误符号是 ".1<5")
0

[root@server ~]# ((-f /etc/passwd)) ?&& echo 1 || echo 0
-bash: ((: -f /etc/passwd:除以 0 (错误符号是 "etc/passwd")
0

1.3 文件测试

可使用以下命令的参数:test? ,[? ?]? ? [[? ]]

参数作用
-b? ? ?文件名检测文件是否是块设备文件,是返回 true
-c? ? ?文件名是否是字符设备文件
-d? ? ?文件名是否是目录
-f? ? ?文件名是否是普通文件 (不是目录,也不是设备文件)
-S? ? 文件名是否为socket文件
-P? ? 文件名是否为管道符文件
-L? ? 文件名是否为链接文件
-u? ? 文件名是否有suid的权限
-s? ? 文件名是否为空(文件大小是否大于0) ,不为空返回 true
-e? ? 文件名检测文件 (包括目录)是否存在,如果是,则返回 true
-r? ? 文件名检测文件是否可读,如果是,则返回 true。
-w? ? 文件名检测文件是否可写,如果是,则返回 true
-X? ? 文件名检测文件是否可执行,如果是,则返回 true
-n? ? 参数检测数字是否为整数
f1 -nt f2文件f1比文件f2则为真 (根据文件修改时间计算)
f1 -ot f2文件f1比文件f2则为真 (根据文件修改时间计算)

例:

[root@server ~]# [ -e /etc/passwd ] ?&& echo yes || echo no
yes
[root@server ~]# [ -f /etc/passwd ] ?&& echo yes || echo no
yes
[root@server ~]# [ -d /etc/passwd ] ?&& echo yes || echo no
no
[root@server ~]# [ -d /etc ] ?&& echo yes || echo no
yes
[root@server ~]# [ -s /etc/passwd ] ?&& echo yes || echo no
yes
[root@server ~]# [ -x /bin/over ] && echo yes || echo no????????
yes

# d开头的均为块文件。c 为字符串文件, l开头为链接文件

则有:

[root@server ~]# [ -c /dev/sr0 ] && echo yes || echo no?
no
[root@server ~]# [ -b /dev/sr0 ] && echo yes || echo no?
yes?

[root@server ~]# [ -L /dev/cdrom ] && echo yes || echo no?
yes? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? # 光盘的快捷方式
[root@server ~]# touch /etc/a.txt
[root@server ~]# [ -s /etc/a.txt ] && echo yes || echo no

no
[root@server ~]# echo ?"hello world" > /etc/a.txt?
[root@server ~]# [ -s /etc/a.txt ] && echo yes || echo no

yes

读,写,执行:

[root@server ~]# ll /etc/a.txt?
-rw-r--r--. 1 root root 12 12月 10 16:30 /etc/a.txt
[root@server ~]# [ -r /etc/a.txt ] && echo yes || echo no
yes
[root@server ~]# [ -w /etc/a.txt ] && echo yes || echo no
yes
[root@server ~]# [ -x /etc/a.txt ] && echo yes || echo no
no

f1 -nt f2:比较的主要是最近改动时间

例题:编写脚本,测试根目录文件是否存在,文件名从键盘输入,若不存在则创建

方法一:使用read命令来获取变量名

????????[root@server ~]# vim test1.sh

#!/bin/bash
read -p "请输入文件名称:"  filename
if test -e /$filename     
then
        echo "$filename exists and it is a normal file"
else
        echo "$filename doesn't exit!!"
        echo "create file $filename"
        touch /$filename
fi    

????????执行查看/下是否存在mnt,并且查看nmt是否存在,若不存在则创建?

方法二:使用位置变量来获取变量名

?????????[root@server ~]# vim test1.sh

#!/bin/bash
filename=$1
if test -e /$filename
then
        echo "$filename exists and it is a normal file"
else
        echo "$filename doesn't exit!!"
        echo "create file $filename"
        touch /$filename
fi

??执行查看/下是否存在mnt,并且查看mnnt是否存在,若不存在则创建

注意:使用find检索文件,无论成功与否状态码均为一

1.4.整数测试

1.4.1.作用

????????用于比较两个数值的大小关系,操作的对象是数值

1.4.2.操作符

整数二元比较操作符使用参考

在[? ]以及 test 中使用的比较符号在(())和[[ ]] 中使用的比较符号说明
-eq==或=相等,全拼为 equal
-ne!=不相等,全拼为 not equal
-gt>大于,全拼为 greater than
-ge>=大于等于,全拼为 greater equal
-lt<小于,全拼为 less than
-le<=小于等于,全拼为 less equal
#知识扩展
#=~ :? ? ?检查左侧内容是否匹配右侧的正则表达式

1.4.3.示例:

[root@server ~]# [ 6 -gt 3 ] && echo yes || echo no
yes
[root@server ~]# [ 6 -lt 3 ] && echo yes || echo no
no
[root@server ~]# test ?6 -lt 3 ?&& echo yes || echo no
no
[root@server ~]# test ?6 -gt 3 ?&& echo yes || echo no
yes
[root@server ~]# [[ 6 < 3 ]] && echo yes || echo no
no
[root@server ~]# [[ 6 > 3 ]] && echo yes || echo no
yes
[root@server ~]# ((6>3)) && echo yes || echo no
yes
[root@server ~]# ((6<3)) && echo yes || echo no
no
[root@server ~]# ((6 < 3)) && echo yes || echo no
no

????????#=~ :? ? ?检查左侧内容是否匹配右侧的正则表达式

[root@server ~]# n1=123
[root@server ~]# [[ $n1 =~ ^[0-9]+$ ]] && echo yes || echo no

yes? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?# 匹配n1内的内容是否全为数字

[root@server ~]# n1=123abd
[root@server ~]# [[ $n1 =~ ^[0-9]+$ ]] && echo yes || echo no

no
[root@server ~]# n1="china"
[root@server ~]# [[ $n1 =~ ^[0-9]+$ ]] && echo yes || echo no

no

例题一:编写表达式,测试系统用户以50个为界限输出相关信息。(多余或少于)

[root@server ~]# [[ $(cat /etc/passwd |wc -l) > 50 ]] && ?echo "用户大于50" || echo "用户小于50"?
用户小于50

例题二:判断当前的已登录的账户数,超过5个则输出提示信息

方法一:

[root@server ~]# ?who |cut -d " " -f1 | sort -u | wc -l? ?# 过滤登陆用户个数
1

#?who:查看账户信息? ?

# cut -d? " " -f1: 以空格作为列间隔符,截取第1列

# sort -u : 去重后排序

[root@server ~]# [ $(who |cut -d " " -f1 | sort -u | wc -l) -lt 5 ] && echo "登陆用户数小于5" || echo "登陆用户数大于5"?
登陆用户数小于5

方法二:

[root@server ~]# ?num=$(who |cut -d " " -f1 | sort -u | wc -l) ?
[root@server ~]# [ $num -lt 5 ] && echo "登陆用户数小于5" || echo "登陆用户数大于5"?
登陆用户数小于5

例三:如果/var/log/messages 文件行数大于50行,则显示提示信息

方法一:

????????[root@server ~]# mnu=$(cat /var/log/messages | wc -l) ?
????????[root@server ~]# echo $mnu

????????102
????????[root@server ~]# [ $mnu -gt 50 ] && echo "文件行数大于50" || echo "文件行数小于50"
????????文件行数大于50

方法二:

????????[root@server ~]# [ $(cat /var/log/messages | wc -l) -lt 50 ] && echo "文件行数小于50" || echo "文件行数大于50"?
????????文件行数大于50
????????[root@server ~]# [ $(cat /var/log/messages | wc -l) -gt 50 ] && echo "文件行数大于50" || echo "文件行数小于50"?
????????文件行数大于50

????????[root@server ~]# (( $(cat /var/log/messages | wc -l) > 50 )) && echo "文件行数大于50" || echo "文件行数小于50"?
????????文件行数大于50
注意:

????????[root@server ~]# [[ $(cat /var/log/messages | wc -l) > 50 ]] && echo "文件行数大于50" || echo "文件行数小于50"?
????????文件行数小于50? ? ? ? ? ? ? ?

????????# 答案错误,原因:双方括号?[[ ]],它是 Bash shell 特有的条件判断符号。在这个命令中,$(cat /var/log/messages | wc -l)?的计算方式与第一个命令相同。然而,条件判断符号?>?在双方括号中被解释为字符串的比较而不是数字的比较,因此结果是错误的。

例三:编写一个脚本mkfs.sh,功能: 显示root账户下的文件信息,之后建立一个aa目录,在aa目录中新建一个文件bb.txt,并修改为可执行的权限

[root@server ~]# vim mkfs.sh

# Author: xx
# Email: xx@163.com
# Organization: http://www.xx.com/xx/
# Created Time : 2023-12-10 18:40:07
# Description:
##############################################################
ls /root
mkdir /root/aa
touch /root/aa/bb.sh
chmod a+x /root/aa/bb.txt

?[root@server ~]# bash mkfs.sh?

例四: 编写脚本mkd.sh,从键盘输入一个目录名,判断是否存在,不存在则创建

方法一:

????????[root@server ~]# vim mkd.sh

# 任务:编写脚本mkd.sh,从键盘输入一个目录名,判断是否存在,不存在则创建

read -p "请输入一个目录名称:" dir

[ -d /$dir  ] && ls -l $dir && echo "目录$dir存在"  || echo "$dir目录不存在,创建目录$dir" && mkdir /$dir

ls /

????????测试:查看根目录下是否有haha目录

方法二:

#!/bin/bash
# 任务:编写脚本mkd.sh,从键盘输入一个目录名,判断是否存在,不存在则创建

read -p "请输入一个目录名称:" dir

[ -d /$dir  ] && ls -l $dir && echo "目录$dir存在"  || (echo "$dir目录不存在,创建目录$dir" ; mkdir /$dir;ls -l /)

注意:???";"?链接括号中多个子语句,从左到右一次执行??

????????测试:查看根目录下是否有haha目录

例5: 编写脚本sum1.sh,从键盘读入俩个整数存储到x,y中,计算其和值后输出

方法一:缺点:每输入一次(无论对错)就要执行一次.sh文件)

read -p "输入整数x:" x
read -p "输入整数y:" y
[[ $x =~ ^[0-9]+$ ]] && [[ $y =~ ^[0-9]+$ ]] && echo "输出整数x+y的结果;$[x+y]" || echo ">请输入数字"

方法二:

read -p "输入整数x:" x
read -p "输入整数y:" y
if [ -n $x -a -n $y ]        # 判断是否为空
then
        if [[ $x =~ ^[0-9]+$ && $y =~ ^[0-9]+$ ]]        # 判断是否为数字
        then
                sum=$[x+y]
                echo "俩参数和为$sum"
        else
                echo "请输入数字"
        fi
else
        echo "请输入有效数字"

fi

例6: 编写脚本user1.sh,显示所有账户,选择一个账户输入,若id为0则表示管理员,否则表示其他账户

首先取出账户名:paswd文件的第列

????????[root@server ~]# cat /etc/passwd ?| cut -d ":" -f1

去重并排序:

????????[root@server ~]# cat /etc/passwd ?| cut -d ":" -f1 | sort -u

[root@server ~]# vim user2.sh

#!/bin/bash

cat /etc/passwd |cut -d ":" -f1 | sort -u
read -p "请输入账户名" name
[ $(id -u $name) -eq 0 ]  && echo "该账户为管理员" || echo "此账户为普通用户"

????????例7: 编写脚本line1.sh,统计/etc/sos/sos.conf文件中的空白行,若有空白行则显示行数,否则输出提示信息(空白行:行首行尾以及中间没有内容"^$"来表示空白行)

[root@server ~]# grep ?^$ /etc/sos/sos.conf |wc -l
4? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? # 输出文件中的空白行 (使用正则表达式中的^:行首,$:行尾表示。)

?[root@server ~]# vim line1.sh

#!bin/bash
num=$( grep ^$ /etc/sos/sos.conf |wc -l )
        
if [ $num -gt 0 ]
then    
        echo "/etc/sos/sos.conf有$num行"
        echo "行好为:"
        grep -n ^$ /etc/sos/sos.conf
else    
        echo “该文件没有空白行”
fi

扩展:将以上服务业务扩展到任何可读文件?

read -p "输入文件路径:" filename

num=$( grep ^$ /$filename |wc -l )

if [ $num -gt 0 ]
then
        echo "/$filename有$num行"
        echo "行号为:"
        grep -n ^$ /$filename
else
        echo “该文件没有空白行”
fi

1.5.逻辑操作符

1.5.1.符号

逻辑操作符

在[]和 test 中使用的操作符在[[ ]] 和(())中使用的操作符说明
-a&&and,与,两端都为真,则结果为真
-o||or,或,两端有一个为真,则结果为真
!!not,非,两端相反,则结果为真

1.5.2.例:

root@server ~]# ?[ -f /etc/passwd -a -f /etc/services ] ?&& echo 1 || echo 0
1
[root@server ~]# ?[ -f /etc/passwd -o -f /etc/services ] ?&& echo 1 || echo 0
1

[root@server ~]# ?[ -f /etc/passwd -a -d /etc/services ] ?&& echo 1 || echo 0
0
[root@server ~]# ?[ -f /etc/passwd -o -d /etc/services ] ?&& echo 1 || echo 0
1

1.6.命令分隔符

cmd1;cmd2? ? ? ? ? ? ? ? ? ? ? 以独立的进程依次执行cmd1和cmd2

(cmd1;cmd2)? ? ? ? ? ? ? ? ? ?在同一进程中依次执行cmd1和cmd2

cmd1&cmd2? ? ? ? ? ? ? ? ? ? cmd1和cmd2同时执行,分属于不同的进程????????

cmd1&&cmd2????????????????当cmd1为真时,则执行cmd2????????

cmd1llcmd2? ? ? ? ? ? ? ? ? ?当cmd1不为真时,则执行cmd2

cmd&? ? ? ? ? ? ? ? ? ? ? ? ? ? ?后台执行

例:

判断一个账户是否存在,若账户fox10不存在,则添加账户

????????[root@server ~]# id fox10 &> /dev/null && echo "fox10已存在" || useradd fox10

? ? ? ? ? ? ? ? ? ? ? ? #??第一次执行命令无返回值,原因,系统判断无fox10因此添加账户ox10
????????[root@server ~]# id fox10 &> /dev/null && echo "fox10已存在" || useradd fox10

????????fox10已存在? ? ? # 由于上一次执行命令时以添加fox10因此显示已存在

注意:? ?&>? /dev/null? 表示将左侧命令执行的正确(或错误)输出到“黑洞”即不显示到屏幕

文章来源:https://blog.csdn.net/2302_77035737/article/details/134895075
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。