![apple_guet apple_guet]()
表格 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)进程替换进程替换 双圆括号 (( var = 78 ))整型运算var=$(( 20 + 5 ))整型运算, 并将结果赋值给变量 引号 "$variable""弱"引用'string'"强"引用 后置引用 result=`COMMAND`在子shell中运行命令, 并将结果赋值给变量--------------------------------START_TIME=`date +%s`系统开机到现在经过的秒数local passed_secs=$((`date +%s` - $START_TIME))其实可以直接使用$SECONDS--------------------------------shell的function返回数值,比如function test_func() { return 255 # return 256}如果return 255,那么echo $?等于255如果return 256,那么echo $?等于0所以$?只占一个字节的空间,函数返回数据最大不能超过255,而且函数只能返回整数,不能返回字符串--------------------------------如果$2参数没有输入,那么设置SECONDS默认5秒[ -z $2 ] && SECONDS=5--------------------------------查询输入的参数为0-9的数字字符串echo "$2" | grep -q '[^0-9]' && Usage && exit 1--------------------------------$RANDOM: 产生随机整数$RANDOM是Bash的内部函数 (并不是常量), 这个函数将返回一个伪随机 [1] 整数, 范围在0 - 32767之间. 它不应该被用来产生密匙.我们可以重新分配随机数种子:RANDOM=1 赋值操作就是产生一个新的随机数种子.RANDOM=$$ 当前脚本进程ID作为随机数种子break命令可以带一个参数. 一个不带参数的break命令只能退出最内层的循环, 而break N可以退出N层循环.continue N结构如果用在有意义的场合中, 往往都很难理解, 并且技巧性很高.所以最好的方法就是尽量避免使用它.《摘自:http://www.tsnc.edu.cn/default/tsnc_wgrj/doc/abs-3.9.1_cn/html/loopcontrol.html》{if ($1 == "'"$module"'") n = "yes";}"'"表示输入一个'单引号所以分解为"'"$module"'"exec 9while read line; do echo 'luther.gliethttp '$line; done--------------------------------1.遍历~/目录下的所有文件夹和文件for FILE in ~/* ; do echo $FILE ; done所有隐藏文件和目录for FILE in ~/.* ; do echo $FILE ; doneluther@gliethttp:~$ for FILE in ~/run.sh/* ; do echo $FILE ; done/home/luther/run.sh/*如果对文件执行操作,那么将直接返回/home/luther/run.sh/*luther@gliethttp:~$ for FILE in ~/run.sh/.* ; do echo $FILE ; done/home/luther/run.sh/.*case "aaa" in "ac") echo 'aaaa' ;; "ac") echo 'ccc' ;; *) echo 'bbb' ;; esaccase可以有相同的数据,但是将只是匹配第一个出现的case数据值下的内容.*)类似于default:关键字'$1/*' | '$1/.' | '$1/..')这里|表示或is_empty_dir() { for FILE in $1/* $1/.* do case "$FILE" in "$1/.*") return 0 ;; "$1/*"|"$1/."|"$1/..") continue ;; *) return 1 ;; esac done return 0}--------------------------------execexec > outfile 默认从lib库中的文件描述符1输出exec > outfile 2> errfile 将2描述符erro信息输出到文件errfileexec 3$ read line1Alice JonesAlice Jones$ (read line1; echo $line1; read line2; echo $line2)Alice JonesRobert Smith$ exec 3$ read -u3 line1; echo $line1; read -u3 line2; echo $line2Alice JonesRobert Smith$ exec 3$ to_screen1 > out 2> errmessage to the user$ cat outmessage to standard output$ cat errluther@gliethttp:~$ exec 9Helponmodule struct:NAMEluther@gliethttp:~$ cat struct.cHelp on module struct:NAME structFILE /usr/lib/python2.5/struct.pyMODULE DOCS http://www.python.org/doc/current/lib/module-struct.html所以read可以根据tab,空格等来进一步细分,读取一行中的某几个域,这在ubuntu7.10的/lib/init/mount-functions.sh是一个很好的学习范例.-------------------------------- if [ -f /etc/fstab ] then exec 9 while read TAB_DEV TAB_MTPT TAB_FSTYPE TAB_OPTS TAB_REST do case "$TAB_DEV" in (""|/#*) continue ;; esac [ "$MTPT" = "$TAB_MTPT" ] || continue [ "$FSTYPE" = "$TAB_FSTYPE" ] || continue case "$TAB_OPTS" in noauto|*,noauto|noauto,*|*,noauto,*) exec 0 return ;; ?*) OPTS="-o$TAB_OPTS" ;; esac break done exec 0 ficase "$TAB_DEV" in (""|/#*) continue ;; esac""表示空行,如果改行只有一个回车或者tab空格等,字符串将等于空""|表示或/#*表示以#号开头的内容/#表示符号#*表示所有字符noauto|*,noauto|noauto,*|*,noauto,*)分为几段:noauto或者*,noauto或者noauto,*或者*,noauto,*都可以使case "$TAB_OPTS" in 成立 12-22 20:53