Linux Bash编程须要注意的几个地方

平常开发基本都在linux环境下,bash命令用的很多,可是写脚本的地方很少。真的写起来,发现须要注意的语法、使用习惯的小问题仍是很多。linux

查了些资料,也结合本身工做中遇到的一些问题,列举几个须要注意的地方shell

1. 变量及赋值

用等号给变量赋值,=先后都不能有空格。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
2. 重定向

重定向:将命令的结果输出到文件,而不是标准输出(屏幕)。
code

  > 写入文件并覆盖旧文件orm

  >> 加到文件的尾部,保留旧文件内容。three

3. 单引号/双引号/反引号

单引号 '进程

            两个单引号包围起来的字符串就是普通的字符串,它将保留原始的字面意思。ip

双引号"

            两个双引号包围起来的字符串,部分特殊字符将起到它们的做用.

            这些特殊字符有: 美圆符$, 反斜杠\, 反引号',  感叹号!.

反引号 `         

            两个反引号包围起来的字符串,将做为命令来运行,
            执行的输出结果做为该反引号的内容,称为命令替换,

            它有另外一种更好的写法: $(command)

tar -zcvf lastmod.tar.gz `find . -mtime -1 -type f -print`
4. if 控制语句

if [ ... ] then

...

elif ... then

...

else

...

fi

用" [ ] "来表示条件测试。注意这里的空格很重要!要确保方括号的空格。

[ -f "somefile" ]  :判断是不是一个文件

[ -x "/bin/ls" ]    :判断/bin/ls是否存在并有可执行权限

[ -n "$var" ]       :判断$var变量是否有值

[ "$a" == "$b" ]   :判断$a和$b是否相等

5. 循环控制

-eq 等于 
-ne 不等于 
-gt 大于 
-ge 大于等于 
-lt 小于 

-le 小于等于


6. 使用{}加强变量的功能

? : 变量未定义,则显示提示语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
7. Bash中的特殊变量:位置变量等

位置变量:$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

相关文章
相关标签/搜索