2010-8
2
Linux Shell 脚本编程基础
本文作者:Leal
授权许可:
创作共享协议
GNU 自由文档许可证
编辑者:谋万世全局者
我们可以使用任意一种文字编辑器,比如 gedit、kedit、emacs、vi 等来编写 shell
脚本,它必须以如下行开始(必须放在文件的第一行):
# !/bin/sh
…
注意:最好使用“!/bin/bash”而不是“!/bin/sh”,如果使用 tc shell 改为 tcsh,其他
类似。
符号#!用来告诉系统执行该脚本的程序,本例使用/bin/sh。编辑结束并保存后,如
果要执行该脚本,必须先使其可执行:
chmod +x filename
此后在该脚本所在目录下,输入 ./filename 即可执行该脚本。
目录
1 变量赋值和引用
2 Shell 里的流程控制
o
o
o
o
o
2.1 if 语 句
2.2 && 和 || 操作符
2.3 case 语句
2.4 select 语句
2.5 while/for 循环
o
o
o
3 Shell 里的一些特殊符号
3.1 引号
4 Here Document
5 Shell 里的函数
6 Shell 脚本示例
6.1 二进制到十进制的转换
6.2 文件循环拷贝
7 脚本调试
变量赋值和引用
Shell 编程中,使用变量无需事先声明,同时变量名的命名须遵循如下规则:
1. 首个字符必须为字母(a-z,A-Z)
2. 中间不能有空格,可以使用下划线(_)
3. 不能使用标点符号
4. 不能使用 bash 里的关键字(可用 help 命令查看保留关键字)
需要给变量赋值时,可以这么写:
变量名=值
要取用一个变量的值,只需在变量名前面加一个$ ( 注意: 给变量赋值的时候,不能
在”=”两边留空格 )
#!/bin/sh
# 对变量赋值:
a=”hello world” #等号两边均不能有空格存在
# 打印变量 a 的值:
echo “A is:” $a
挑个自己喜欢的编辑器,输入上述内容,并保存为文件 first,然后执行 chmod +x
first 使其可执行,最后输入 ./first 执行该脚本。其输出结果如下:
A is: hello world
有时候变量名可能会和其它文字混淆,比如:
num=2
echo “this is the $numnd”
上述脚本并不会输出”this is the 2nd”而是”this is the “;这是由于 shell 会去搜索变
量 numnd 的值,而实际上这个变量此时并没有值。这时,我们可以用花括号来告
诉 shell 要打印的是 num 变量:
num=2
echo “this is the ${num}nd”
其输出结果为:this is the 2nd
注意花括号的位置:
num=2
echo “this is the {$num}nd”
其输出结果为:this is the {2}nd
需要注意 shell 的默认赋值是字符串赋值。比如:
var=1
var=$var+1
echo $var
打印出来的不是 2 而是 1+1。为了达到我们想要的效果有以下几种表达方式:
let “var+=1″
var=$[$var+1]
var=`expr $var + 1`#注意加号两边的空格,否则还是按照字符串的方式赋值。
注意:前两种方式在 bash 下有效,在 sh 下会出错。
let 表示数学运算,expr 用于整数值运算,每一项用空格隔开,$[]将中括号内的表
达式作为数学运算先计算结果再输出。
Shell 脚本中有许多变量是系统自动设定的,我们将在用到这些变量时再作说明。除
了只在脚本内有效的普通 shell 变量外,还有环境变量,即那些由 export 关键字处
理过的变量。本文不讨论环境变量,因为它们一般只在登录脚本中用到。
Shell 里的流程控制
if 语 句
“if”表达式如果条件为真,则执行 then 后的部分:
if ….; then
….
elif ….; then
….
else
….
fi
大多数情况下,可以使用测试命令来对条件进行测试,比如可以比较字符串、判断
文件是否存在及是否可读等等……通常用” [ ] “来表示条件测试,注意这里的空格很
重要,要确保方括号前后的空格。
[ -f "somefile" ] :判断是否是一个文件
[ -x "/bin/ls" ] :判断/bin/ls 是否存在并有可执行权限
[ -n "$var" ] :判断$var 变量是否有值
[ "$a" = "$b" ] :判断$a 和$b 是否相等
执行 man test 可以查看所有测试表达式可以比较和判断的类型。下面是一个简单的
if 语句:
#!/bin/sh
if [ ${SHELL} = "/bin/bash" ]; then
echo “your login shell is the bash (bourne again shell)”
else
echo “your login shell is not bash but ${SHELL}”
fi
变量$SHELL 包含有登录 shell 的名称,我们拿它和/bin/bash 进行比较以判断当前
使用的 shell 是否为 bash。
&& 和 || 操作符
熟悉 C 语言的朋友可能会喜欢下面的表达式:
[ -f "/etc/shadow" ] && echo “This computer uses shadow passwords”
这里的 && 就是一个快捷操作符,如果左边的表达式为真则执行右边的语句,你也
可以把它看作逻辑运算里的与操作。上述脚本表示如果/etc/shadow 文件存在,则
打印”This computer uses shadow passwords”。同样 shell 编程中还可以用或操作
(||),例如:
#!/bin/sh
mailfolder=/var/spool/mail/james
[ -r "$mailfolder" ] || { echo “Can not read $mailfolder” ; exit 1; }
echo “$mailfolder has mail from:”
grep “^From ” $mailfolder
该脚本首先判断 mailfolder 是否可读,如果可读则打印该文件中的”From” 一行。
如果不可读则或操作生效,打印错误信息后脚本退出。需要注意的是,这里我们必
须使用如下两个命令:
-打印错误信息
-退出程序
我们使用花括号以匿名函数的形式将两个命令放到一起作为一个命令使用;普通函
数稍后再作说明。即使不用与和或操作符,我们也可以用 if 表达式完成任何事情,
但是使用与或操作符会更便利很多 。
case 语句
case 表达式可以用来匹配一个给定的字符串,而不是数字(可别和 C 语言里的
switch…case 混淆)。
case … in
…) do something here
esac
file 命令可以辨别出一个给定文件的文件类型,如:file lf.gz,其输出结果为:
lf.gz: gzip compressed data, deflated, original filename,
last modified: Mon Aug 27 23:09:18 2001, os: Unix
我们利用这点写了一个名为 smartzip 的脚本,该脚本可以自动解压 bzip2, gzip 和
zip 类型的压缩文件:
#!/bin/sh
ftype=`file “$1″` # Note ‘ and ` is different
case “$ftype” in
“$1: Zip archive”*)
unzip “$1″ ;;
“$1: gzip compressed”*)
gunzip “$1″ ;;
“$1: bzip2 compressed”*)
bunzip2 “$1″ ;;
*) echo “File $1 can not be uncompressed with smartzip”;;
esac
你可能注意到上面使用了一个特殊变量$1,该变量包含有传递给该脚本的第一个参
数值。也就是说,当我们运行:
smartzip articles.zip
$1 就是字符串 articles.zip。
select 语句
========================================
select 表达式是 bash 的一种扩展应用,擅长于交互式场合。用户可以从一组不同的
值中进行选择:
select var in … ; do
break;
done
…. now $var can be used ….
下面是一个简单的示例:
#!/bin/sh
echo “What is your favourite OS?”
select var in “Linux” “Gnu Hurd” “Free BSD” “Other”; do
break;
done
echo “You have selected $var”
如果 以上脚本运行出现 select :NOT FOUND 将 #!/bin/sh 改为 #!/bin/bash
该脚本的运行结果如下:
What is your favourite OS?
1) Linux
2) Gnu Hurd
3) Free BSD
4) Other
#? 1
You have selected Linux
while/for 循环
在 shell 中,可以使用如下循环:
while …; do
….
done
只要测试表达式条件为真,则 while 循环将一直运行。关键字”break”用来跳出循环,
而关键字”continue”则可以跳过一个循环的余下部分,直接跳到下一次循环中。
for 循环会查看一个字符串列表(字符串用空格分隔),并将其赋给一个变量:
for var in ….; do
….
done
下面的示例会把 A B C 分别打印到屏幕上:
#!/bin/sh
for var in A B C ; do
echo “var is $var”
done
下面是一个实用的脚本 showrpm,其功能是打印一些 RPM 包的统计信息:
#!/bin/sh
# list a content summary of a number of RPM packages
# USAGE: showrpm rpmfile1 rpmfile2 …
# EXAMPLE: showrpm /cdrom/RedHat/RPMS/*.rpm
for rpmpackage in $*; do
if [ -r "$rpmpackage" ];then
echo “=============== $rpmpackage ==============”
rpm -qi -p $rpmpackage
else
echo “ERROR: cannot read file $rpmpackage”
fi
done
这里出现了第二个特殊变量$*,该变量包含有输入的所有命令行参数值。如果你运
行 showrpm openssh.rpm w3m.rpm webgrep.rpm,那么 $* 就包含有 3 个字符
串,即 openssh.rpm, w3m.rpm 和 webgrep.rpm。
Shell 里的一些特殊符号
引号
在向程序传递任何参数之前,程序会扩展通配符和变量。这里所谓的扩展是指程序
会把通配符(比如*)替换成适当的文件名,把变量替换成变量值。我们可以使用引
号来防止这种扩展,先来看一个例子,假设在当前目录下有两个 jpg 文件:mail.jpg
和 tux.jpg。
#!/bin/sh
echo *.jpg
运行结果为:
mail.jpg tux.jpg
引号(单引号和双引号)可以防止通配符*的扩展:
#!/bin/sh
echo “*.jpg”
echo ‘*.jpg’
其运行结果为:
*.jpg
*.jpg