Shell脚本 学习
来源:互联网 发布:itunes mac 下载路径 编辑:程序博客网 时间:2024/05/30 13:41
在 shell 中的四则运算必须使用 expr 这个指令来辅助。因为这是一个指令,所以如果要将结果指定给变量,必须使用 ` 包起来。请注意,在 + - * / 的二边都有空白,如果没有空白将产生错误:
$ expr 5 -23$ sum=`expr 5 + 10`$ echo $sum15$ sum=`expr $sum / 3`$ echo $sum5
还有一个要特别注意的是乘号 * 在用 expr 运算时,不可只写 *。因为 * 有其它意义,所以要使用 /* 来代表。另外,也可以用 % 来求余数。
$ count=`expr 5 /* 3`$ echo $count$ echo `expr $count % 3`5
我们再列出更多使用 expr 指令的方式,下列表中为可以放在指令 expr 之后的表达式。有的符号有特殊意义,必须以 / 将它的特殊意义去除,例如 /*,否则必须用单引号将它括起来,如 '*':
类别 语法 说明 条件判断 expr1 /| expr2 如果 expr1 不是零或 null 则传回 expr1,否则传回 expr2。 expr1 /& expr2 如果 expr1 及 expr2 都不为零或 null,则传回 expr1,否则传回 0。 四则运算 expr1 + expr2 传回 expr1 加 expr2 后的值。 expr1 - expr2 传回 expr1 减 expr2 后的值。 expr1/* expr2 传回 expr1 乘 expr2 后的值。 expr1 / expr2 传回 expr1 除 expr2 后的值。 expr1 % expr2 传回 expr1 除 expr2 的余数。 大小判断 expr1 /> expr2 如果 expr1 大于 expr2 则传回 1,否则传回 0。如果 expr1 及 expr2 都是数字,则是以数字大小判断,否则是以文字判断。以下皆同。 expr1 /< expr2 如果 expr1 小于 expr2 则传回 1,否则传回 0。 expr1 = expr2 如果 expr1 等于 expr2 则传回 1,否则传回 0。 expr1 != expr2 如果 expr1 不等于 expr2 则传回 1,否则传回 0。 expr1 />= expr2 如果 expr1 大于或等于 expr2 则传回 1,否则传回 0。 expr1 /<= expr2 如果 expr1 小于或等于 expr2 则传回 1,否则传回 0。 文字处理 expr1 : expr2 比较一固定字符串,即 regular expression。可以使用下列字符来辅助:. 匹配一个字符。
$ 找字符串的结尾。
[list] 找符合 list 中的任何字符串。
* 找寻 0 个或一个以上在 * 之前的字。
/( /) 传回括号中所匹配的字符串。
我们针对比较复杂的文字处理部份再加以举例:
$ ttyttyp0$ expr `tty` : ".*/(../)/___FCKpd___2quot;p0$ expr `tty` : '.*/(../)
上面执行 tty 的结果是 ttyp0,而在 expr 中,在 : 右侧的表达式中,先找 .* 表示0个或一个以上任何字符,传回之后在结尾 ($) 时的二个字符 /(../)。在第一个 expr 的式子中,因为使用双引号,所以在 $ 之前要用一个 / 来去除 $ 的特殊意义,而第二个 expr 是使用单引号,在单引号内的字都失去了特殊意义,所以在 $ 之前不必加 /。
除了使用 expr 外,我们还可以使用下列这种特殊语法:
$ a=10$ b=5$ c=$((${a}+${b}))$ echo $c15$ c=$((${a}*${b}))$ echo $c50
我们可以使用 $(()) 将表达式放在括号中,即可达到运算的功能。
表格 B-1. 特殊的shell变量
$0
脚本名字 $1
位置参数 #1 $2 - $9
位置参数 #2 - #9 ${10}
位置参数 #10 $#
位置参数的个数 "$*"
所有的位置参数(作为单个字符串) * "$@"
所有的位置参数(每个都作为独立的字符串) ${#*}
传递到脚本中的命令行参数的个数 ${#@}
传递到脚本中的命令行参数的个数 $?
返回值 $$
脚本的进程ID(PID) $-
传递到脚本中的标志(使用set) $_
之前命令的最后一个参数 $!
运行在后台的最后一个作业的进程ID(PID) * 必须被引用起来, 否则默认为"$@
".
表格 B-2. 测试操作: 二元比较
-eq
等于 =
等于 ==
等于 -ne
不等于 !=
不等于 -lt
小于 /<
小于 (ASCII) * -le
小于等于 -gt
大于 />
大于 (ASCII) * -ge
大于等于 -z
字符串为空 -n
字符串不为空 算术比较 双括号(( ... ))结构 >
大于 >=
大于等于 <
小于 <=
小于等于 * 如果在双中括号 [[ ... ]] 测试结构中使用的话, 那么就不需要使用转义符/了.
表格 B-3. 文件类型的测试操作
-e
文件是否存在 -s
文件大小不为0 -f
是一个标准文件 -d
是一个目录 -r
文件具有读权限 -h
文件是一个符号链接 -w
文件具有写权限 -L
文件是一个符号链接 -x
文件具有执行权限 -b
文件是一个块设备 -c
文件是一个字符设备 -g
设置了sgid标记 -p
文件是一个管道 -u
设置了suid标记 -S
文件是一个socket -k
设置了"粘贴位" -t
文件与一个终端相关联 -N
从这个文件最后一次被读取之后, 它被修改过 F1 -nt F2
文件F1比文件F2新 * -O
这个文件的宿主是你 F1 -ot F2
文件F1比文件F2旧 * -G
文件的组id与你所属的组相同 F1 -ef F2
文件F1和文件F2都是同一个文件的硬链接 * !
"非" (反转上边的测试结果) * 二元操作符(需要两个操作数).
表格 B-4. 参数替换和扩展
${var}
变量var
的值, 与$var
相同 ${var-DEFAULT}
如果var
没有被声明, 那么就以$DEFAULT
作为其值 * ${var:-DEFAULT}
如果var
没有被声明, 或者其值为空, 那么就以$DEFAULT
作为其值 * ${var=DEFAULT}
如果var
没有被声明, 那么就以$DEFAULT
作为其值 * ${var:=DEFAULT}
如果var
没有被声明, 或者其值为空, 那么就以$DEFAULT
作为其值 * ${var+OTHER}
如果var
声明了, 那么其值就是$OTHER
, 否则就为null字符串 ${var:+OTHER}
如果var
被设置了, 那么其值就是$OTHER
, 否则就为null字符串 ${var?ERR_MSG}
如果var
没被声明, 那么就打印$ERR_MSG
* ${var:?ERR_MSG}
如果var
没被设置, 那么就打印$ERR_MSG
* ${!varprefix*}
匹配之前所有以varprefix
开头进行声明的变量 ${!varprefix@}
匹配之前所有以varprefix
开头进行声明的变量 * 当然, 如果变量var
已经被设置的话, 那么其值就是$var
.
表格 B-5. 字符串操作
${#string}
$string
的长度 ${string:position}
在$string
中, 从位置$position
开始提取子串 ${string:position:length}
在$string
中, 从位置$position
开始提取长度为$length
的子串 ${string#substring}
从变量$string
的开头, 删除最短匹配$substring
的子串 ${string##substring}
从变量$string
的开头, 删除最长匹配$substring
的子串 ${string%substring}
从变量$string
的结尾, 删除最短匹配$substring
的子串 ${string%%substring}
从变量$string
的结尾, 删除最长匹配$substring
的子串 ${string/substring/replacement}
使用$replacement
, 来代替第一个匹配的$substring
${string//substring/replacement}
使用$replacement
, 代替所有匹配的$substring
${string/#substring/replacement}
如果$string
的前缀匹配$substring
, 那么就用$replacement
来代替匹配到的$substring
${string/%substring/replacement}
如果$string
的后缀匹配$substring
, 那么就用$replacement
来代替匹配到的$substring
expr match "$string" '$substring'
匹配$string
开头的$substring
*的长度 expr "$string" : '$substring'
匹配$string
开头的$substring
*的长度 expr index "$string" $substring
在$string
中匹配到的$substring
的第一个字符出现的位置 expr substr $string $position $length
在$string
中从位置$position
开始提取长度为$length
的子串 expr match "$string" '/($substring/)'
从$string
的开头位置提取$substring
* expr "$string" : '/($substring/)'
从$string
的开头位置提取$substring
* expr match "$string" '.*/($substring/)'
从$string
的结尾提取$substring
* expr "$string" : '.*/($substring/)'
从$string
的结尾提取$substring
* * $substring
是一个正则表达式.
表格 B-6. 一些结构的汇总
if [ CONDITION ]
测试结构 if [[ CONDITION ]]
扩展的测试结构 Array[1]=element1
数组初始化 [a-z]
正则表达式的字符范围 大括号 ${variable}
参数替换 ${!variable}
间接变量引用 { command1; command2; . . . commandN; }
代码块 {string1,string2,string3,...}
大括号扩展 圆括号 ( command1; command2 )
子shell中执行的命令组 Array=(element1 element2 element3)
数组初始化 result=$(COMMAND)
在子shell中执行命令, 并将结果赋值给变量 >(COMMAND)
进程替换 <(COMMAND)
进程替换 双圆括号 (( var = 78 ))
整型运算 var=$(( 20 + 5 ))
整型运算, 并将结果赋值给变量 引号 "$variable"
"弱"引用 'string'
"强"引用 后置引用 result=`COMMAND`
在子shell中运行命令, 并将结果赋值给变量 上面执行 tty 的结果是 ttyp0,而在 expr 中,在 : 右侧的表达式中,先找 .* 表示0个或一个以上任何字符,传回之后在结尾 ($) 时的二个字符 /(../)。在第一个 expr 的式子中,因为使用双引号,所以在 $ 之前要用一个 / 来去除 $ 的特殊意义,而第二个 expr 是使用单引号,在单引号内的字都失去了特殊意义,所以在 $ 之前不必加 /。
除了使用 expr 外,我们还可以使用下列这种特殊语法:
___FCKpd___3
我们可以使用 $(()) 将表达式放在括号中,即可达到运算的功能。