平常开发基本都在linux环境下,bash命令用的很多,可是写脚本的地方很少。真的写起来,发现须要注意的语法、使用习惯的小问题仍是很多。linux
查了些资料,也结合本身工做中遇到的一些问题,列举几个须要注意的地方shell
用等号给变量赋值,=先后都不能有空格。bash
export能够将其设为环境变量,当前SHELL及子进程有效;父进程的自定义变量没法在子进程中使用。测试
var=32 now=`date +%y%m%d%H` cur_path=`dirname $0` PATH=$PATH:/home/user/bin
declare -x var也可讲变量设为环境变量。变量赋值,默认是string类型,经过declare命令可改为其余类型,好比:spa
declare -i sum=12+34 #i = integer declare -a ... #a = array
重定向:将命令的结果输出到文件,而不是标准输出(屏幕)。
code
> 写入文件并覆盖旧文件orm
>> 加到文件的尾部,保留旧文件内容。three
单引号 '进程
两个单引号包围起来的字符串就是普通的字符串,它将保留原始的字面意思。ip
双引号"
两个双引号包围起来的字符串,部分特殊字符将起到它们的做用.
这些特殊字符有: 美圆符$, 反斜杠\, 反引号', 感叹号!.
反引号 `
两个反引号包围起来的字符串,将做为命令来运行,
执行的输出结果做为该反引号的内容,称为命令替换,
它有另外一种更好的写法: $(command)
tar -zcvf lastmod.tar.gz `find . -mtime -1 -type f -print`
if [ ... ] then
...
elif ... then
...
else
...
fi
用" [ ] "来表示条件测试。注意这里的空格很重要!要确保方括号的空格。
[ -f "somefile" ] :判断是不是一个文件
[ -x "/bin/ls" ] :判断/bin/ls是否存在并有可执行权限
[ -n "$var" ] :判断$var变量是否有值
[ "$a" == "$b" ] :判断$a和$b是否相等
-eq 等于
-ne 不等于
-gt 大于
-ge 大于等于
-lt 小于
-le 小于等于
? : 变量未定义,则显示提示语word,脚本执行结束;
- : 变量未定义, 则使用新值word; 不改变变量原来的状态;
+: 变量定义, 则使用新值word; 不改变变量原来的状态; 与?不一样的是,脚本不会中止执行;
=: 变量未定义, 则使用新值word, 并定义到变量;
变量有未定义与空值之分,要同时对这两种状况都进行判断,能够在上面的方法中加入冒号: 增长功能:
例如:
$ # a is undefined $ b="" $ c="Z" $ echo a=${a-1}, b=${b-2}, c=${c-3} a=1, b=, c=Z $ echo a=${a:-1}, b=${b:-2}, c=${c:-3} a=1, b=2, c=Z
位置变量:$1~$9, 表明传入给执行脚本的参数,从1开始,最多能够有9个。例如,下面的脚本根据输入的名字实现重命名功能:
#!/bin/sh # rename: - rename a file # Usage: rename oldname newname oldname=$1 newname=$2 mv ${oldname:?"missing"} ${newname:?"missing"}
若是传入的参数超过9个,或者说想传入任意数量的参数,能够用shift命令,来移除传入的参数,例如:
#!/bin/sh arg1=$1;shift; arg2=$1;shift; arg3=$1;shift; echo first three arguments are $arg1 $arg2 and $arg3
#!/bin/sh arg1=$1 arg2=$2 arg3=$3;shift 3 echo first three arguments are $arg1 $arg2 and $arg3
$0 - 脚本名字
$* - 全部的位置参数
使用这个变量,一样能够操做超过10个的位置参数。例以下面这个脚本,能够把任意多个的文件移动到一个文件夹下:
#!/bin/sh # scriptname: moveto # usage: # moveto directory files..... directory=${1:?"Missing"};shift mv $* $directory 调用的方法: ./moveto.sh /mytmp *.txt
$@ - 全部的位置参数(参数可包含空格)
此变量与$*的基本功能相似,不一样的地方是,@能够处理包含空格的参数。空格能够分隔输入参数,若是输入参数包含空格,$@能够返回包含空格的参数,$*则不能。能够参考下面的例子来理解:
首先有一个打印变量名的脚本EchoArgs
#!/bin/sh # ================ # usage: # echoes arguments # ================ E="echo -n" # echo the name of script ${E} $0: # now echo each argument ${E} "'${1-"?"}'" ${E} "'${2-"?"}'" ${E} "'${3-"?"}'" ${E} "'${4-"?"}'" ${E} "'${5-"?"}'" ${E} "'${6-"?"}'" ${E} "'${7-"?"}'" echo
再写一个测试脚本TestEchoArgs.sh
#!/bin/sh # ================ # usage: # test echoes arguments # ================ ./EchoArgs $* ./EchoArgs $@ ./EchoArgs "$*" ./EchoArgs "$@"
测试输入结果,比较两个命令的不一样:
./TestEchoArgs.sh "a b c" 'd e' f g 输出: ./EchoArgs:'a''b''c''d''e''f''g' ./EchoArgs:'a''b''c''d''e''f''g' ./EchoArgs:'a b c d e f g''?''?''?''?''?''?' ./EchoArgs:'a b c''d e''f''g''?''?''?'
分析:
./EchoArgs $* ./EchoArgs $@ 前两个没放在""中,结果同样,由于$*和$@的执行以后都是传给EchoArgs由空格分隔的七个参数 ./EchoArgs "$*" 因为$*不能分辨有空格的参数,最后传给EchoArgs的只有一个""的参数 ./EchoArgs "$@" $@在""中分辨出有四个参数,前两个参数中有空格。
$# - 参数个数
shift $#能够去掉全部的传入参数。
$$ - 当前进程的ID
因为进程的ID都是不一样的,这个变量能够用来命名惟一的临时文件。例以下面这个脚本,返回行数:
#!/bin/sh # ************************************** # usage: # count lines of input file/files # ************************************** tempfile=/tmp/$0.$$ for i in $@ do if [ -f $i ]; then cat $i >> $tempfile fi done echo `wc -l $tempfile` lines were found /bin/rm $tempfile
另外一个用处是,能够在其余进程中经过$$来kill这个进程。例如:
echo $$ >/tmp/job.pid kill -HUP `cat /tmp/job.pid`
$! - 后台进程的PID
只有运行&命令的进程才会改变$!.使用此命令能够控制多个后台运行的程序,好比:
#!/bin/sh job1 & pid=$! job2 & pid="$pid $!" job3 & pid="$pid $!" trap "kill -15 $pid" 0 1 2 15 wait
$? - 错误状态
返回前一个程序的退出状态。
----------
未完,不定时更新...
Last updated - 2/23/2016