Shell里的数组
Shell里的数组类似python里的tuple ,使用小括号标识。并且元素之间直接使用空格分隔就行了,举个例子:
# 错误定义,这样相当于定义了一个元素为‘1,2,3’的数组。
Huper@Huper:/usr/local/src/learningShell$ array=(1,2,3)
Huper@Huper:/usr/local/src/learningShell$ echo $array
1,2,3
# 错误引用方式,必须加上大括号。
Huper@Huper:/usr/local/src/learningShell$ echo $array[0]
1,2,3[0]
Huper@Huper:/usr/local/src/learningShell$ echo ${array[0]}
1,2,3
# 正确用法。
Huper@Huper:/usr/local/src/learningShell$ array=(1 2 3)
Huper@Huper:/usr/local/src/learningShell$ echo $array
1
Huper@Huper:/usr/local/src/learningShell$ echo ${array[0]}
1
还记得如何获取所有命令行参数和参数长度吗?这里同样适用:
Huper@Huper:/usr/local/src/learningShell$ echo ${array[*]}
1 2 3
Huper@Huper:/usr/local/src/learningShell$ echo ${array[@]}
1 2 3
Huper@Huper:/usr/local/src/learningShell$ echo ${#array[*]}
3
运算符
因为毕竟不能与编程语言划等号,所以原生的bash一般不支持直接的表达式计算,它能理解的都是一些命令。比如说bash里有expr这个表达式命令,可以把需要计算的表达式作为参数传递给它,来看以下这些情况:
Huper@Huper:/usr/local/src/learningShell$ 1 + 1
1: command not found
Huper@Huper:/usr/local/src/learningShell$ expr 1+1
1+1
Huper@Huper:/usr/local/src/learningShell$ expr 1 + 1
2
可以看到bash对表达式的要求也很严格,必须有空格分开。怎么保存这个值呢:
Huper@Huper:/usr/local/src/learningShell$ plus=`expr 1 + 1`
Huper@Huper:/usr/local/src/learningShell$ echo $plus
2
是不是很鬼畜?除了这些之外,运算符的使用方式和意义基本与常用编程语言相似了,下表列出了常用的算术运算符(注意乘号是保留符号,表示全部,所以必须转义):
| 运算符 | 说明 | 举例 | 
|---|---|---|
| + | 加法 | 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。 | 
然后是关系运算符,除了等于和不等于,必须使用一系列以-开头的标识:
| 运算符 | 说明 | 举例 | 
|---|---|---|
| -eq | 检测两个数是否相等,相等返回 true。 | [ a -eq b ] 返回 false。 | 
| -ne | 检测两个数是否相等,不相等返回 true。 | [ a -ne b ] 返回 true。 | 
| -gt | 检测左边的数是否大于右边的,如果是,则返回 true。 | [ a -gt b ] 返回 false。 | 
| -lt | 检测左边的数是否小于右边的,如果是,则返回 true。 | [ a -lt b ] 返回 true。 | 
| -ge | 检测左边的数是否大于等于右边的,如果是,则返回 true。 | [ a -ge b ] 返回 false。 | 
| -le | 检测左边的数是否小于等于右边的,如果是,则返回 true。 | [ a -le b ] 返回 true。 | 
逻辑运算比较简单:
| 运算符 | 说明 | 举例 | 
|---|---|---|
| ! | 非运算,表达式为 true 则返回 false,否则返回 true。 | [ ! false ] 返回 true。 | 
| -o/|| | 或运算,有一个表达式为 true 则返回 true。 | [ a -lt 20 -o b -gt 100 ] 返回 true。 | 
| -a/&& | 与运算,两个表达式都为 true 才返回 true。 | [ a -lt 20 -a b -gt 100 ] 返回 false。 | 
然后是字符串操作,Shell里的字符串运算符还是蛮多的:
| 运算符 | 说明 | 举例 | 
|---|---|---|
| = | 检测两个字符串是否相等,相等返回 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。 | 
最后是文件相关的运算符,这个就比较实用了:
| -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。 | 
以上五组表格里,除了第一组运算操作(不包括最后两个),剩下的四组运算操作实际上都返回逻辑值,可以配合Shell里的条件语句if使用,例子中之所以用[]把表达式括起来,也是因为if的语法要求。关于if和其他流程控制语句,留到下一次再讲。