shell(shell变量、条件表达式、流程控制)

本章内容:

  1. 变量
  2. 运算
  3. if语句
  4. for语句
  5. while语句
  6. break、continue
  7. 实例

shell变量

1、shell变量简介

变量是任何一种编程语言都必不可少的组成部分,变量用来存放各种数据。脚本语言在定义变量时通常不需要指明类型,直接赋值就可以,Shell 变量也遵循这个规则。

在 Bash shell 中,每一个变量的值都是字符串,无论你给变量赋值时有没有使用引号,值都会以字符串的形式存储;这意味着,Bash shell 在默认情况下不会区分变量类型,即使你将整数和小数赋值给变量,它们也会被视为字符串,这一点和大部分的编程语言不同。

2、定义变量

Shell 常用三种定义变量的方式:

variable=value
variable='value'    #所见即所得
variable="value"   #进行转义
variable=`cat test`  #执行命令将命令输出赋值给变量  

Shell 变量的命名规范和大部分编程语言都一样:

  • 变量名由数字、字母、下划线组成;
  • 必须以字母或者下划线开头;
  • 不能使用 Shell 里的关键字(通过 help 命令可以查看保留关键字)。

3、使用变量

╭─root@localhost.localdomain ~  
╰─➤  aa="cjk"
╭─root@localhost.localdomain ~  
╰─➤  echo $aa
cjk
╭─root@localhost.localdomain ~  
╰─➤  echo ${aa}   #推荐给所有变量加上花括号{ },这是个良好的编程习惯
cjk

4、删除变量

使用 unset 命令可以删除变量

╭─root@localhost.localdomain ~  
╰─➤  aa="cjk"
╭─root@localhost.localdomain ~  
╰─➤  echo ${aa}
cjk
╭─root@localhost.localdomain ~  
╰─➤  unset aa   #unset 命令不能删除只读变量
╭─root@localhost.localdomain ~  
╰─➤  echo ${aa}

5、变量类型

运行shell时,会同时存在三种变量:

局部变量

局部变量在脚本或命令中定义,仅在当前shell实例中有效,其他shell启动的程序不能访问局部变量。

环境变量

所有的程序,包括shell启动的程序,都能访问环境变量,有些程序需要环境变量来保证其正常运行。必要的时候shell脚本也可以定义环境变量。

shell变量

shell变量是由shell程序设置的特殊变量。shell变量中有一部分是环境变量,有一部分是局部变量,这些变量保证了shell的正常运行

6、系统变量

在命令行提示符直接执行 env、set 查看系统或环境变量。env 显示用户环境变量,set 显示 Shell 预先定义好的变量以及用户变量。可以通过 export 导出成用户变量。一些写 Shell 脚本时常用的系统变量

系统变量名 系统变量意识
$SHELL 默认 Shell
$HOME 当前用户家目录
$IFS 内部字段分隔符
$LANG 默认语言
$PATH 默认可执行程序路径
$PWD 当前目录
$UID 当前用户 ID
$USER 当前用户
$HISTSIZE 历史命令大小,可通过 HISTTIMEFORMAT 变量设置命令执行时间
$RANDOM 随机生成一个 0 至 32767 的整数
$HOSTNAME 主机名

生成8位随机数:echo "$RANDOM"|md5sum|cut -c 1-8

7、普通变量和环境变量

  • 普通变量定义:VAR=value
  • 临时环境变量定义:export VAR=value
  • 变量引用:$VAR
  • 区别:Shell 进程的环境变量作用域是 Shell 进程,当 export 导入到系统变量时,则作用域是 Shell 进程及其 Shell 子进程,另开shell无效。

8、特殊变量列表(shell变量)

变量 含义
$0 当前脚本的文件名
$n 传递给脚本或函数的参数。n 是一个数字,表示第几个参数。例如,第一个参数是$1,第二个参数是$2。
$# 传递给脚本或函数的参数个数。
$* 传递给脚本或函数的所有参数。
$? 上个命令的退出状态,或函数的返回值,成功会返回 0,失败返回非0
$$ 当前Shell进程ID,对于 Shell 脚本,就是这些脚本所在的进程ID

实例:

╭─root@localhost.localdomain ~  
╰─➤  vim test.sh
...
#!/bin/bash

echo "File Name: $0"

echo "First Parameter : $1"

echo "First Parameter : $2"

echo "Quoted Values: $@"

echo "Quoted Values: $*"

echo "Total Number of Parameters : $#"
...
╭─root@localhost.localdomain ~  
╰─➤  bash test.sh  hello cjk touch
File Name: test.sh
First Parameter : hello
First Parameter : cjk
Quoted Values: hello cjk touch
Quoted Values: hello cjk touch
Total Number of Parameters : 3
╭─root@localhost.localdomain ~  
╰─➤  echo $?                      
0
╭─root@localhost.localdomain ~  
╰─➤  ddd ddd
zsh: command not found: ddd
╭─root@localhost.localdomain ~  
╰─➤  echo $?    
127

shift 命令 将参数向前移一位
shift n 向前移n位


shell条件表达式与运算符

9、条件表达式

表达式 实例
[ expression ] [ 1 -eq 1 ]
[[ expression ]] [[ 1 -eq 1 ]]
test expression test 1 -eq 1 ,等同于[]

注意:括号中的表达式前后都有空格,否则会报错!

╭─root@localhost.localdomain ~  
╰─➤  [ 1 -eq 1 ] && echo "true" || echo "false"
true
╭─root@localhost.localdomain ~  
╰─➤  [ 1 -eq 2 ] && echo "true" || echo "false"
false
╭─root@localhost.localdomain ~  
╰─➤  [1 -eq 2] && echo "true" || echo "false" 
zsh: bad pattern: [1

10、整数比较符

比较符 描述 实例
-eq, equal 等于 [ 1 -eq 1 ] 为true
-ne, not equal 不等于 [ 1 -ne 1 ] 为false
-gt, greate than 大于 [ 1 -gt 1 ] 为false
-lt, lesser than 小于 [ 1 -lt 1 ] 为false
-ge, greate or equal 大于或者等于 [ 1 -ge 1 ] 为true
-le, lesser or equal 小于或者等于 [ 1 -le 1 ] 为true

11、字符串比较符

运算符 描述 实例
== 等于 [ “a” == “a” ] 为true
!= 不等于 [ “a” != “a” ] 为false
-n 字符串长度不等于 0 为真 VAR1=1;VAR2=””
[ -n “$VAR1” ]为 true
[ -n “$VAR2” ]为 false
-z 字符串长度等于 0 为真 VAR1=1;VAR2=””
[ -z “$VAR1” ]为false
[ -z “$VAR2” ]为 true

注意:使用-n 判断字符串长度时,变量要加双引号,养成好习惯,字符串比较时都加上双引号

12、文件测试

测试符 描述 实例
-e 文件或者目录存在为真 [ -e path ] path 存在为 true
-f 文件存在为真 [ -f file_path ] 文件存在为 true
-d 目录存在为真 [ -d dir_path ] 目录存在为 true
-r 有读权限为真 [ -r file_path ]file_path有读权限为真
-w 有写权限为真 [ -w file_path ]file_path有写权限为真
-x 有执行权限为真 [ -x file_path ]file_path有执行权限为真
-s 文件存在且不为空为真 [-s file_path]file_path存在且不为空为真

13、布尔运算符

运算符 描述 实例
非关系,条件结果取反 [ ! 1 -eq 2 ]为true
-a 和关系,在[]表达式中使用 [ 1 -eq 1 -a 2 -eq 2 ]为true
两者都为真才为真
-o 或关系,在[]表达式中使用 [ 1 -eq 1 -o 2 -eq 1 ]为true
两者有一真则为真

14、逻辑判断符

判断符 描述 实例
&& 逻辑和,在[[]]表达式中或判断表达式是否为真时使用 [[ 1 -eq 1 && 2 -eq 2 ]]为 true
[ 1 -eq 1 ] && echo ‘true’
如果&&前面的表达式为true则执行后面的
|| 逻辑或,在[[]]表达式中或判断表达式是否为真时使用 [[ 1 -eq 1 || 2 -eq 1 ]]为 true
[ 1 -eq 2 ] || echo ‘true’
如果||前面的表达式为false则执行后面的

15、整数运算符

运算符 描述
+ 加法
减法
* 乘法
/ 除法
% 取余
运算表达式 实例
$(()) $((1+1))
$[] $[]

16、其他运算符

命令 描述 实例
let 赋值并运算 let x++;echo $x 每执行一次 x 加 1
let y–;echo $y 每执行一次 y 减 1
let x+=2 每执行一次 x 加 2
let x-=2 每执行一次 x 减 2
expr 乘法*需要\转义"\*" expr 1 \* 2 运算符两边必须有空格
expr\( 1 + 2\) \* 2 使用双括号时要转义

if语句

if语句用exit结束

1.1单分支

if 条件表达式 ;then

    命令

fi

实例:

#!/bin/bash
read -p "请输入数字" num
if [ $num -lt 10 ] ;then
        echo "${num}是数字两位数"
fi
...

╭─root@localhost.localdomain ~  
╰─➤  bash test.sh
请输入数字3
3是数字一位数

1.2、多分支

if 条件表达式 ;then
    命令
else
    命令
fi

实例:判断read输出的数字是两位数

#!/bin/bash
read -p "请输入数字" num
if [ $num -gt 9 -a $num -lt 100 ] ;then
        echo "${num}是数字两位数"
else
        echo "${num}不是数字两位数"
fi
...

╭─root@localhost.localdomain ~  
╰─➤  bash test.sh 
请输入数字56
56是数字两位数
╭─root@localhost.localdomain ~  
╰─➤  bash test.sh
请输入数字222
222不是数字两位数

1.3、多分支

if 条件表达式 ;then
    命令
elif 条件表达式 ;then
    命令
else
    命令
fi

实例:

#!/bin/bash
num=`echo $RANDOM`
if [ $num -lt 1000 ];then
        echo "$num小于1000"
elif [ $num -ge 1000 -a $num -lt 2000 ];then
        echo "$num大于等于1000,小于2000"
elif [ $num -ge 2000 -a $num -lt 3000 ];then
        echo "$num大于等于2000,小于3000"
elif [ $num -ge 3000 -a $num -lt 4000 ];then
        echo "$num大于等于3000,小于4000"
else
        echo "$num大于等于4000"
fi
...
╭─root@localhost.localdomain ~  
╰─➤  bash test.sh
24497大于等于4000
╭─root@localhost.localdomain ~  
╰─➤  bash test.sh
20763大于等于4000

for 循环

for  变量名  in  取值列表

do

    命令

done

实例:循环ping主机并输出是否在线

#!/bin/bash
. /etc/init.d/functions
ip=192.168.80.
for i in `seq 10`
do
        if ping -c 1 -w 1 $ip$i &>/dev/null;then
                echo -n " $ip$i"    #-n:作用不换行  ;$ip前有空格
                success
                echo " "
                sleep 1
        else
                echo -n " $ip$i"
                failure
                echo " "
                sleep 1
        fi
done

...
╭─root@localhost.localdomain ~  
╰─➤  bash test.sh
 192.168.80.1                                              [  OK  ]
 192.168.80.2                                              [  OK  ]
 192.168.80.3                                              [  OK  ]
 192.168.80.4                                              [  OK  ]
 192.168.80.5                                              [FAILED]
 192.168.80.6                                              [FAILED]
 192.168.80.7                                              [FAILED]
 192.168.80.8                                              [FAILED]
 192.168.80.9                                              [FAILED]
 192.168.80.10                                             [FAILED]
------------------------------------------------------------------------------
#没有functions文件的系统可以如下

[root@k8s-master duzhuo]# cat ping2.sh
#!/bin/bash
#. /etc/init.d/functions
ip=192.168.75.20
for i in `seq 9`
do
        if ping -c 1 -w 1 $ip$i &>/dev/null;then
                echo -n " $ip$i"    #-n:作用不换行  ;$ip前有空格
                echo -e "\t [\033[32m success \033[0m]"
                sleep 1
        else
                echo -n " $ip$i"
                echo -e "\t [\033[5;31m failed \033[0m]"     #不在线机器红色闪烁
                sleep 1
        fi
done

-----

[root@k8s-master duzhuo]# sh ping2.sh
 192.168.75.201  [ success ]
 192.168.75.202  [ success ]
 192.168.75.203  [ failed ]
 192.168.75.204  [ failed ]
 192.168.75.205  [ failed ]
 192.168.75.206  [ failed ]
 192.168.75.207  [ failed ]
 192.168.75.208  [ failed ]
 192.168.75.209  [ failed ]

while循环

while 条件表达式 ; do

      命令

done

3.1、一般循环

当条件表达式为 false 时,终止循环

实例:

#!/bin/bash
N=0
while [ $N -lt 5 ]; do
        echo $N
        let N++
done
...
╭─root@localhost.localdomain ~  
╰─➤  bash test1.sh
0
1
2
3
4

3.2、死循环

条件表达式为 true,将会产生死循环

#!/bin/bash
while true ;do
        echo "hello"
        sleep 1
done

...
╭─root@localhost.localdomain ~  
╰─➤  bash test2.sh
hello
hello
hello
hello
hello
^C
  • 命令行中输入 nohup bash naolie.sh & 即可在后台持续运行该脚本

break和continue语句

  • continue 与 break 语句只能循环语句中使用;
  • break:终止循环,运行针跳至done后;
  • continue :跳出当前循环,运行针跳至do后,开始下一循环;

演示:
break语句

#!/bin/bash
N=0
while [ $N -lt 10 ]; do
        let N++
        if [ $N -eq 5 ]; then
                break
        fi
        echo $N
done
...
╭─root@localhost.localdomain ~  
╰─➤  bash test1.sh
1
2
3
4

continue语句

#!/bin/bash
N=0
while [ $N -lt 10 ]; do
        let N++
        if [ $N -eq 5 ]; then
                continue
        fi
        echo $N
done
...
╭─root@localhost.localdomain ~  
╰─➤  bash test1.sh
1
2
3
4
6
7
8
9
10

实例拓展:

实例1:巡检系统主机名ip系统版本等

#!/bin/bash
echo "主机名:`hostname`"
echo "IP地址:`ip a | grep "global" | cut -d "/" -f 1 | cut -d "t" -f 2 | tr -d " "`"
echo "操作系统版本:`cat /etc/redhat-release`"
echo "内核版本:`uname -r`"
echo "CPU信息:`lscpu | grep -i "Model name" | cut -c "24-69"`"
echo "内存总大小:`free -h | grep "Mem" | cut -d "M" -f 2 | tr -d "em: "`M"
------------------------------------------
╭─root@localhost.localdomain ~  
╰─➤  bash test6.sh
主机名:localhost.localdomain
IP地址:192.168.80.3
操作系统版本:CentOS Linux release 7.3.1611 (Core) 
内核版本:3.10.0-514.el7.x86_64
CPU信息:Intel(R) Core(TM) i5-4210M CPU @ 2.60GHz
内存总大小:976M

实例二:统计.sh结尾的文件总大小

#!/bin/bash
sum=0
for i in `find / -type f -a -name "*.sh"`
do
        size=`ls -l $i | cut -d " " -f 5`
        let sum+=size
done
echo ".sh结尾的总大小$(($sum/1024))kb"
...
╭─root@localhost.localdomain ~  
╰─➤  bash test7.sh
.sh结尾的总大小1098kb

实例三:创建100个用户并设置6位随机密码

[root@du ~]# cat test9.sh
#!/bin/bash
#!/bin/bash
for i in `seq 10`
do
    useradd user$i
    pass=`echo $RANDOM | md5sum | cut -c 1-6`
    echo "$pass" | passwd --stdin "user$i"
    echo -e "账户:user$i\n密码:$pass" >> /root/passwd
done

##执行结果:

[root@du ~]# cat passwd
账户:user1
密码:69a70a
账户:user2
密码:444c02
账户:user3
密码:6b381f
账户:user4
密码:28d8fd

实例四:找出100以内的质数

#!/bin/bash
for i in `seq 100`
do
    for((j=2;j<i;j++))
    do
         [ $((i%j)) -eq 0 ] &&  break
    done
         [ $j -eq $i ] &&   echo $i
done


## 执行
╭─root@localhost.localdomain ~  
╰─➤  bash test.sh 
2
3
5
7
11
13
17
19
23
29
31
37
41
43
47
53
59
61
67
71
73
79
83
89
97

实例五:逐行读取文件

#!/bin/bash
cat $1 | while read line
do
echo “$line”
sleep 1
done

## 执行

╭─root@localhost.localdomain ~  
╰─➤  bash readtest.sh /etc/passwd
“root:x:0:0:root:/root:/bin/zsh”
“bin:x:1:1:bin:/bin:/sbin/nologin”
“daemon:x:2:2:daemon:/sbin:/sbin/nologin”
“adm:x:3:4:adm:/var/adm:/sbin/nologin”
“lp:x:4:7:lp:/var/spool/lpd:/sbin/nologin”
^C#  

实例六:shell脚本工整化输出,表格形式输出

摘自:https://blog.csdn.net/qq_41228463/article/details/80720772

#!/bin/bash
table=""
#设置行,可以是表头,也可以是表格内容。
#如果是表格内容,“—”表示空值
function setRow(){
    value=$*
    table=${table}"|${value// /#|}#|\n"
}

#行分隔线
#入参:表格的列数。如表格有5列,则入参为5
function splitLine(){
    local num=`expr $1 + 2`
    split=`seq -s '+#' $num | sed 's/[0-9]//g'`    # 生成连续个的+#
    table=${table}"${split}\n"
}

#绘制表格
#入参:table
function setTable(){
    echo -e $1|column -s "#" -t|awk '{if($0 ~ /^+/){gsub(" ","-",$0);print $0}else{print $0}}'
}
if [ "$SHLVL" -eq "2" ]
then
   table=""
    splitLine 3
    setRow "姓名" "性别" "年龄"
    splitLine 3
    setRow "Tom" "male" "10"
    setRow "Nacy" "female" "-"
    splitLine 3
    setTable ${table}
fi
##输出结果

#bash test.sh
+------+--------+------+--
|姓名  |性别    |年龄  |
+------+--------+------+--
|Tom   |male    |10    |
|Nacy  |female  |-     |
+------+--------+------+--

posted @ 2019-06-01 16:07  du-z  阅读(9291)  评论(0编辑  收藏  举报