Shell脚本学习2
运算符
Bash 支持很多运算符,包括算数运算符、关系运算符、布尔运算符、字符串运算符和文件测试运算符。
算术运算符
运算符 说明 举例 + 加法 `expr $a + $b` 结果为 30。 - 减法 `expr $a - $b` 结果为 10。 * 乘法 `expr $a \* $b` 结果为 200。 / 除法 `expr $b / $a` 结果为 2。 % 取余 `expr $b % $a` 结果为 0。 = 赋值 a=$b 将把变量 b 的值赋给 a。 == 相等。用于比较两个数字,相同则返回 true。 [ $a == $b ] 返回 false。 != 不相等。用于比较两个数字,不相同则返回 true。 [ $a != $b ] 返回 true。
原生bash不支持简单的数学运算,但是可以通过其他命令来实现,例如 awk 和 expr,expr 最常用。
expr
是一款表达式计算工具,使用它能完成表达式的求值操作。
# 命令行直接计算 expr 2 + 2 #4 expr 3 - 2 #1 expr 3 / 2 #1 expr 3 \* 2 #6 # 使用表达式 a=10 b=20 val=`expr $a + $b` echo "a + b : $val"
注意:
- 表达式和运算符之间要有空格
- 乘号(*)前边必须加反斜杠\才能实现乘法运算
- 完整的表达式要被 ` ` 包含
关系运算符
运算符 说明 -eq 检测两个数是否相等,相等返回 true。同算数运算符`==` -ne 检测两个数是否相等,不相等返回 true -gt 检测左边的数是否大于右边的,如果是,则返回 true。 -lt 检测左边的数是否小于右边的,如果是,则返回 true。 -ge 检测左边的数是否大等于右边的,如果是,则返回 true。 -le 检测左边的数是否小于等于右边的,如果是,则返回 true。
关系运算符只支持数字,不支持字符串,除非字符串的值是数字。
#!/bin/sh a=10 b=20 if [ $a -eq $b ] then echo "$a -eq $b : a is equal to b" else echo "$a -eq $b: a is not equal to b" fi
布尔运算符
布尔运算符列表
运算符 说明 ! 非运算,表达式为 true 则返回 false,否则返回 true。 -o 或运算(or),有一个表达式为 true 则返回 true。 -a 与运算(and),两个表达式都为 true 才返回 true。
字符串运算符
字符串运算符列表
运算符 说明 举例 = 检测两个字符串是否相等,相等返回 true。 [ $a = $b ] 返回 false。 != 检测两个字符串是否相等,不相等返回 true。 [ $a != $b ] 返回 true。 -z 检测字符串长度是否为0,为0返回 true。 [ -z $a ] 返回 false。 -n 检测字符串长度是否为0,不为0返回 true。 [ -n $a ] 返回 true。 str 检测字符串是否为空,不为空返回 true。 [ $a ] 返回 true。
文件测试运算符
文件测试运算符用于检测 Unix 文件的各种属性。
操作符 说明 举例 -b file 检测文件是否是块设备文件,如果是,则返回 true。 [ -b $file ] 返回 false。 -c file 检测文件是否是字符设备文件,如果是,则返回 true。 [ -c $file ] 返回 false。 -d file 检测文件是否是目录,如果是,则返回 true。 [ -d $file ] 返回 false。 -f file 检测文件是否是普通文件(既不是目录,也不是设备文件),如果是,则返回 true。 [ -f $file ] 返回 true。 -g file 检测文件是否设置了 SGID 位,如果是,则返回 true。 [ -g $file ] 返回 false。 -k file 检测文件是否设置了粘着位(Sticky Bit),如果是,则返回 true。 [ -k $file ] 返回 false。 -p file 检测文件是否是具名管道,如果是,则返回 true。 [ -p $file ] 返回 false。 -u file 检测文件是否设置了 SUID 位,如果是,则返回 true。 [ -u $file ] 返回 false。 -r file 检测文件是否可读,如果是,则返回 true。 [ -r $file ] 返回 true。 -w file 检测文件是否可写,如果是,则返回 true。 [ -w $file ] 返回 true。 -x file 检测文件是否可执行,如果是,则返回 true。 [ -x $file ] 返回 true。 -s file 检测文件是否为空(文件大小是否大于0),不为空返回 true。 [ -s $file ] 返回 true。 -e file 检测文件(包括目录)是否存在,如果是,则返回 true。 [ -e $file ] 返回 true。
#!/bin/sh file="/tmp/test.sh" if [ -e $file ] then echo "File exists" else echo "File does not exist" fi
字符串
字符串是shell编程中最常用最有用的数据类型(除了数字和字符串,也没啥其它类型好用了),字符串可以用单引号,也可以用双引号,也可以不用引号。
单双引号的区别:
- 双引号里可以有变量,单引号则原样输出;
- 双引号里可以出现转义字符,单引号则原样输出;
- 单引号字串中不能出现单引号。
拼接字符串
#!/bin/bash str1='i' str2='love' str3='you' echo $str1 $str2 $str3 echo $str1$str2$str3 echo $str1,$str2,$str3 //i love you //iloveyou //i,love,you
获取字符串长度
#!/bin/bash/ str='i love you' echo ${#str} # 输出:10
截取字符串
#!/bin/bash/ str='i love you' echo ${str:1} # 从第1个截取到末尾。注意从0开始。 echo ${str:2:2} # 从第2个截取2个。 echo ${str:0} # 全部截取。 echo ${str:-3} # 负数无效,视为0。 echo ${str::3} # 截取[0,3)
查找字符串
#!/bin/bash/ str="i love you" echo `expr index "$str" l` echo `expr index "$str" you` #最后一个参数是字符集,返回最靠前字符的index echo `expr index "$str" o` echo `expr length "$str"` #字符串长度 echo `expr substr "$str" 1 6` #从第1个字符开始截取6个字符
拓展:expr
更多关于字符串用法:
STRING : REGEXP #anchored pattern match of REGEXP in STRING match STRING REGEXP #same as STRING : REGEXP substr STRING POS LENGTH #从STRING中POS位置开始截取LENGTH个字符。POS索引是从1开始的。 index STRING CHARS #在STRING中查找字符CHARS首次出现的位置,没有找到返回0 length STRING #字符串长度
数组
bash支持一维数组(不支持多维数组),并且没有限定数组的大小,数组元素的下标由0开始编号
获取数组中的元素要利用下标,下标可以是整数或算术表达式,其值应大于或等于0。
定义数组的一般形式为:
array_name=(value1 value2 ... valuen) array_name=( value0 value1 value2 value3 )
注:用括号来表示数组,数组元素用空格
符号分割开
单独赋值数组元素:
array_name[0]=value0 array_name[1]=value1 array_name[2]=value2
读取数组:
echo ${array_name[2]} #读取下标为2的元素 echo ${array_name[*]} #读取所有元素 echo ${array_name[@]} #读取所有元素 echo ${#array_name[*]} #获取数组长度 echo ${#array_name[@]} #获取数组长度 echo ${#array_name[1]} #获取数组中单个元素的长度
条件控制
if语句
语法格式:
if [ expression ] then Statement(s) to be executed if expression is true fi
注意:expression
和方括号([ ])之间必须有空格,否则会有语法错误。
Shell 有三种 if … else 语句:
if ... fi 语句 if ... else ... fi 语句 if ... elif ... else ... fi 语句
if ... else
语句也经常与 test
命令结合使用:
#!/bin/bash/ if test $a == $b then echo "a is equal to b" else echo "a is not equal to b" fi
case语句
case ... esac
与其他语言中的 switch ... case
语句类似,是一种多分枝选择结构。
#!/bin/bash/ grade="B" case $grade in "A") echo "Very Good!";; "B") echo "Good!";; "C") echo "Come On!";; *) echo "You Must Try!" echo "Sorry!";; esac
说明:
取值后面必须为关键字 in
每一模式必须以右括号结束
取值可以为变量或常数
匹配发现取值符合某一模式后,其间所有命令开始执行直至 ;;
如果无一匹配模式,使用星号 *
捕获该值
#!/bin/bash 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
for循环
语法格式:
for 变量 in 列表 do command1 command2 ... commandN done
顺序输出字符串中的字符:
for str in 'This' 'is' 'a' 'string' do echo $str done
遍历目录下的文件:
#!/bin/bash for FILE in * do echo $FILE done
遍历文件内容:
#!/bin/bash citys=`cat city.txt` for city in $citys do echo $city done
while循环
只要while后面的条件满足,就一直执行do里面的代码块。
语法格式为:
while command do Statement(s) to be executed if command is true done
示例:
#!/bin/bash c=0; while [ $c -lt 3 ] do echo "Value c is $c" c=`expr $c + 1` #c变量自增 done
until循环
until 循环执行一系列命令直至条件为 true 时停止。until 循环与 while 循环在处理方式上刚好相反。
#!/bin/bash c=0; until [ $c -eq 3 ] do echo "Value c is $c" c=`expr $c + 1` done
跳出循环
Shell也使用 break 和 continue 来跳出循环。
break命令允许跳出所有循环:
#!/bin/bash i=0 while [ $i -lt 5 ] do i=`expr $i + 1` if [ $i == 3 ] then break fi echo -e $i done
在嵌套循环中,break 命令后面还可以跟一个整数,表示跳出第几层循环。例如:
break n
continue:它不会跳出所有循环,仅仅跳出当前循环。
#!/bin/bash i=0 while [ $i -lt 5 ] do i=`expr $i + 1` if [ $i == 3 ] then continue fi echo -e $i done