一个网上的shell参考教材,很好。第 31 章 Shell脚本
http://learn.akae.cn/media/ch31s05.html#id2874366
========================
为何下面的脚本使用 /bin/bash 就不报错,使用/bin/sh就报错?
而且在suse环境下使用/bin/sh也不报错,在ubuntu下就报错?
./1.sh: 9: [[: not found
no
3 s1='12345'
4
5 if [[ $s1 == '12*' ]] ;then
6 echo 'yes'
7 else
8 echo 'no'
9 fi原因在与,[[ ]]是bash的语法,而dash不支持。在ubuntu下,sh是个软链,默认指向了dash,而不是bash!
dash和bash有什么差别呢?为什么ubuntu弃bash而用dash呢?
官方说法是:bash太臃肿,dash小巧。而二进制文件大小上看,dash大小是bash的十分之一。
下面一片文章讲解了bash和dash的差异,
http://princessleia.com/plug/2008-JP_bash_vs_dash.pdf
==========================
shell字符串操作
字符串拼接
a=1234
a+=5678 // now a=12345678
字符串截取 指定位置字串
a=1234567890000
取前8个字符的方法如下
1: expr substr $a 1 8 // 视字符串下标从1开始
2: echo $a|cut -c1-8 // 视字符串下标从1开始
3: echo ${a:0:7} // 视字符串下标从0开始
字符串截取 查找位置字串
${varible##*string} 从左向右截取最后一个string后的字符串
${varible#*string}从左向右截取第一个string后的字符串
${varible%%string*}从右向左截取最后一个string后的字符串
${varible%string*}从右向左截取第一个string后的字符串
“*”只是一个通配符可以不要
例子:
$ MYVAR=foodforthought.jpg
$ echo ${MYVAR##*fo}
rthought.jpg
$ echo ${MYVAR#*fo}
odforthought.jpg
$ echo ${MYVAR%%fo*}
空
$ echo ${MYVAR%fo*}
food
获取文件的后缀名
1:echo ${MYVAR##*.}
2:echo $MYVAR|cut -d. -f2 // -d. 以.分割,-f2第二个分割域
整数比较
-eq 等于,如:if [ "$a" -eq "$b" ]
-ne 不等于,如:if [ "$a" -ne "$b" ]
-gt 大于,如:if [ "$a" -gt "$b" ]
-ge 大于等于,如:if [ "$a" -ge "$b" ]
-lt 小于,如:if [ "$a" -lt "$b" ]
-le 小于等于,如:if [ "$a" -le "$b" ]
< 小于(需要双括号),如:(("$a" < "$b"))
<= 小于等于(需要双括号),如:(("$a" <= "$b"))
> 大于(需要双括号),如:((”$a” > “$b”))
>= 大于等于(需要双括号),如:((”$a” >= “$b”))
字符串比较
http://henry-cong.javaeye.com/blog/677287
字符串比较
= 等于,如:if [ "$a" = "$b" ]
== 等于,如:if [ "$a" == "$b" ],与=等价
注意:==的功能在[[]]和[]中的行为是不同的,如下:
1 [[ $a == z* ]] # 如果$a以”z”开头(模式匹配)那么将为true
2 [[ $a == "z*" ]] # 如果$a等于z*(字符匹配),那么结果为true
3
4 [ $a == z* ] # File globbing 和word splitting将会发生
5 [ "$a" == "z*" ] # 如果$a等于z*(字符匹配),那么结果为true
一点解释,关于File globbing是一种关于文件的速记法,比如”*.c”就是,再如~也是.
但是file globbing并不是严格的正则表达式,虽然绝大多数情况下结构比较像.
!= 不等于,如:if [ "$a" != "$b" ]
这个操作符将在[[]]结构中使用模式匹配.
< 小于,在ASCII字母顺序下.如:
if [[ "$a" < "$b" ]]
if [ "$a" /< "$b" ]
注意:在[]结构中"<"需要被转义.
> 大于,在ASCII字母顺序下.如:
if [[ "$a" > "$b" ]]
if [ "$a" /> "$b" ]
注意:在[]结构中”>”需要被转义.
具体参考Example 26-11来查看这个操作符应用的例子.
-z 字符串为”null”.就是长度为0.
-n 字符串不为”null”
注意:
使用-n在[]结构中测试必须要用”"把变量引起来. 使用一个未被”"的字符串来使用! -z
或者就是未用”"引用的字符串本身,放到[]结构中。虽然一般情况下可
以工作,但这是不安全的. 习惯于使用”"来测试字符串是一种好习惯.
==========================================
下面记录一些有关于shell的砸碎.
一些shell的好的案例.
查找当前目录的工作进程的 正在运行的进程 fuser的实现
#!/bin/bash
#pcwd.sh,查找系统中工作在目标路径下的进程,使用方法: pcwd.sh 路径名
test $1 || { echo "Usage: pcwd.sh path(absolute path)" && exit };
PS=`ls /proc/*[0-9]*|grep :` ;
PSList=`echo $PS |tr -d '/proc'|tr -d ':'`;
for PID in $PSList
do
test -e /proc/$PID && (ls -alF /proc/$PID/cwd|grep $1 >;/dev/null ) && echo "Process $PID (`cat /proc/$PID/status |grep Name |tr -d 'Name:/t'`) is work in $1 ";
done
举例
[elly@frost.llnl]/tmp# ./pcwd.sh
Usage: pcwd.sh path(absolute path)
[elly@frost.llnl]/tmp# ./pcwd.sh /usr
Process 1283 (vi) is work in /usr ;