1、linux 脚本程序编写基础 2012-07-17 15:02:43| 分类: Linux | 标签: linux 开发 linux 笔记 linux |字号 订阅1. Linux 脚本编写基础1.1 语法基本介绍1.1.1 开头程序必须以下面的行开始(必须放在文件的第一行):#!/bin/sh符号#!用来告诉系统它后面的参数是用来执行该文件的程序。在这个例子中我们使用/bin/sh 来执行程序。当编辑好脚本时,如果要执行该脚本,还必须使其可执行。要使脚本可执行:编译 chmod +x filename 这样才能用./filename 来运行1.1.2 注释在进行 shell 编程时,以#开头
2、的句子表示注释,直到这一行的结束。我们真诚地建议您在程序中使用注释。如果您使用了注释,那么即使相当长的时间内没有使用该脚本,您也能在很短的时间内明白该脚本的作用及工作原理。1.1.3 变量在其他编程语言中您必须使用变量。在 shell 编程中,所有的变量都由字符串组成,并且您不需要对变量进行声明。要赋值给一个变量,您可以这样写:#!/bin/sh#对变量赋值:a=“hello world“# 现在打印变量 a 的内容:echo “A is:“echo $a有时候变量名很容易与其他文字混淆,比如:num=2echo “this is the $numnd“这并不会打印出“this is the
3、2nd“,而仅仅打印“this is the “,因为 shell 会去搜索变量numnd 的值,但是这个变量时没有值的。可以使用花括号来告诉 shell 我们要打印的是 num 变量:num=2echo “this is the $numnd“这将打印: this is the 2nd1.1.4 环境变量由 export 关键字处理过的变量叫做环境变量。我们不对环境变量进行讨论,因为通常情况下仅仅在登录脚本中使用环境变量。1.1.5 Shell 命令和流程控制在 shell 脚本中可以使用三类命令:1)Unix 命令:虽然在 shell 脚本中可以使用任意的 unix 命令,但是还是由一些相
4、对更常用的命令。这些命令通常是用来进行文件和文字操作的。常用命令语法及功能echo “some text“: 将文字内容打印在屏幕上ls: 文件列表wc l filewc -w filewc -c file: 计算文件行数计算文件中的单词数计算文件中的字符数cp sourcefile destfile: 文件拷贝mv oldname newname : 重命名文件或移动文件rm file: 删除文件grep pattern file: 在文件内搜索字符串比如:grep searchstring file.txtcut -b colnum file: 指定欲显示的文件内容范围,并将它们输出到标准
5、输出设备比如:输出每行第 5 个到第 9 个字符 cut -b5-9 file.txt 千万不要和 cat 命令混淆,这是两个完全不同的命令cat file.txt: 输出文件内容到标准输出设备(屏幕)上file somefile: 得到文件类型read var: 提示用户输入,并将输入赋值给变量sort file.txt: 对 file.txt 文件中的行进行排序uniq: 删除文本文件中出现的行列比如: sort file.txt | uniqexpr: 进行数学运算 Example: add 2 and 3expr 2 “+“ 3find: 搜索文件比如:根据文件名搜索 find . -
6、name filename -printtee: 将数据输出到标准输出设备(屏幕) 和文件比如:somecommand | tee outfilebasename file: 返回不包含路径的文件名比如: basename /bin/tux 将返回 tuxdirname file: 返回文件所在路径比如: dirname /bin/tux 将返回 /binhead file: 打印文本文件开头几行tail file : 打印文本文件末尾几行sed: Sed 是一个基本的查找替换程序。可以从标准输入(比如命令管道)读入文本,并将结果输出到标准输出(屏幕)。该命令采用正则表达式(见参考)进行搜索。
7、不要和 shell 中的通配符相混淆。比如:将 linuxfocus 替换为LinuxFocus :cat text.file | sed s/linuxfocus/LinuxFocus/ newtext.fileawk: awk 用来从文本文件中提取字段。缺省地,字段分割符是空格,可以使用 -F 指定其他分割符。cat file.txt | awk -F, print $1 “,“ $3 这里我们使用,作为字段分割符,同时打印第一个和第三个字段。如果该文件内容如下: Adam Bor, 34, IndiaKerry Miller, 22, USA命令输出结果为:Adam Bor, India
8、Kerry Miller, USA2) 概念: 管道, 重定向和 backtick这些不是系统命令,但是他们真的很重要。管道 (|) 将一个命令的输出作为另外一个命令的输入。grep “hello“ file.txt | wc -l在 file.txt 中搜索包含有”hello”的行并计算其行数。在这里 grep 命令的输出作为 wc 命令的输入。当然您可以使用多个命令。重定向:将命令的结果输出到文件,而不是标准输出(屏幕)。 写入文件并覆盖旧文件 加到文件的尾部,保留旧文件内容。反短斜线使用反短斜线可以将一个命令的输出作为另外一个命令的一个命令行参数。命令:find . -mtime -1
9、-type f -print用来查找过去 24 小时(-mtime 2 则表示过去 48 小时)内修改过的文件。如果您想将所有查找到的文件打一个包,则可以使用以下脚本:#!/bin/sh# The ticks are backticks () not normal quotes ():tar -zcvf lastmod.tar.gz find . -mtime -1 -type f -print3) 流程控制1.if“if“ 表达式 如果条件为真则执行 then 后面的部分:if .; then.elif .; then.else.fi大多数情况下,可以使用测试命令来对条件进行测试。比如可以比
10、较字符串、判断文件是否存在及是否可读等等通常用“ “来表示条件测试。注意这里的空格很重要。要确保方括号的空格。 -f “somefile“ :判断是否是一个文件 -x “/bin/ls“ :判断/bin/ls 是否存在并有可执行权限 -n “$var“ :判断 $var 变量是否有值 “$a“ = “$b“ :判断$a 和$b 是否相等执行 man test 可以查看所有测试表达式可以比较和判断的类型。直接执行以下脚本:#!/bin/shif “$SHELL“ = “/bin/bash“ ; thenecho “your login shell is the bash (bourne agai
11、n shell)“elseecho “your login shell is not bash but $SHELL“fi变量$SHELL 包含了登录 shell 的名称,我们和/bin/bash 进行了比较。快捷操作符熟悉 C 语言的朋友可能会很喜欢下面的表达式: -f “/etc/shadow“ exit 1; echo “$mailfolder has mail from:“grep “From “ $mailfolder该脚本首先判断 mailfolder 是否可读。如果可读则打印该文件中的“From“ 一行。如果不可读则或操作生效,打印错误信息后脚本退出。这里有个问题,那就是我们必须
12、有两个命令:-打印错误信息-退出程序我们使用花括号以匿名函数的形式将两个命令放到一起作为一个命令使用。一般函数将在下文提及。不用与和或操作符,我们也可以用 if 表达式作任何事情,但是使用与或操作符会更便利很多。2.casecase :表达式可以用来匹配一个给定的字符串,而不是数字。case . in.) do something here ;esac让我们看一个例子。 file 命令可以辨别出一个给定文件的文件类型,比如:file lf.gz这将返回:lf.gz: gzip compressed data, deflated, original filename,last modified:
13、 Mon Aug 27 23:09:18 2001, os: Unix我们利用这一点写了一个叫做 smartzip 的脚本,该脚本可以自动解压 bzip2, gzip 和 zip 类型的压缩文件:#!/bin/shftype=file “$1“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“;e
14、sac您可能注意到我们在这里使用了一个特殊的变量$1。该变量包含了传递给该程序的第一个参数值。也就是说,当我们运行:smartzip articles.zip$1 就是字符串 articles.zip3. selsectselect 表达式是一种 bash 的扩展应用,尤其擅长于交互式使用。用户可以从一组不同的值中进行选择。select var in . ; dobreakdone. now $var can be used .下面是一个例子:#!/bin/shecho “What is your favourite OS?“select var in “Linux“ “Gnu Hurd“ “
15、Free BSD“ “Other“; dobreakdoneecho “You have selected $var“下面是该脚本运行的结果:What is your favourite OS?1) Linux2) Gnu Hurd3) Free BSD4) Other#? 1You have selected Linux4.looploop 表达式:while .; do.donewhile-loop 将运行直到表达式测试为真。will run while the expression that we test for is true.关键字“break“ 用来跳出循环。而关键字”conti
16、nue” 用来不执行余下的部分而直接跳到下一个循环。for-loop 表达式查看一个字符串列表 (字符串用空格分隔) 然后将其赋给一个变量:for var in .; do.done在下面的例子中,将分别打印 ABC 到屏幕上:#!/bin/shfor var in A B C ; doecho “var is $var“done下面是一个更为有用的脚本 showrpm,其功能是打印一些 RPM 包的统计信息:#!/bin/sh# list a content summary of a number of RPM packages# USAGE: showrpm rpmfile1 rpmfil
17、e2 .# EXAMPLE: showrpm /cdrom/RedHat/RPMS/*.rpmfor rpmpackage in $*; doif -r “$rpmpackage“ ;thenecho “= $rpmpackage =“rpm -qi -p $rpmpackageelseecho “ERROR: cannot read file $rpmpackage“fidone这里出现了第二个特殊的变量$*,该变量包含了所有输入的命令行参数值。如果您运行 showrpm openssh.rpm w3m.rpm webgrep.rpm此时 $* 包含了 3 个字符串,即 openssh.rp
18、m, w3m.rpm and webgrep.rpm.5. 引号在向程序传递任何参数之前,程序会扩展通配符和变量。这里所谓扩展的意思是程序会把通配符(比如*)替换成合适的文件名,它变量替换成变量值。为了防止程序作这种替换,您可以使用引号:让我们来看一个例子,假设在当前目录下有一些文件,两个 jpg 文件, mail.jpg 和tux.jpg。1.2 编译 SHELL 脚本#ch#!/bin/sh mod +x filenamecho *.jpg 缓螅梢酝 淙耄?./filename 来执行您的脚本。这将打印出“mail.jpg tux.jpg“的结果。引号 (单引号和双引号) 将防止这种通配符
19、扩展:#!/bin/shecho “*.jpg“echo *.jpg这将打印“*.jpg“ 两次。单引号更严格一些。它可以防止任何变量扩展。双引号可以防止通配符扩展但允许变量扩展。#!/bin/shecho $SHELLecho “$SHELL“echo $SHELL运行结果为:/bin/bash/bin/bash$SHELL最后,还有一种防止这种扩展的方法,那就是使用转义字符反斜杆:echo *.jpgecho $SHELL这将输出:*.jpg$SHELL6. Here documents当要将几行文字传递给一个命令时,here documents(译者注:目前还没有见到过对该词适合的翻译)
20、一种不错的方法。对每个脚本写一段帮助性的文字是很有用的,此时如果我们四有那个 here documents就不必用 echo 函数一行行输出。 一个 “Here document“ 以 shift by 2-) shift;break; # end of options-*) echo “error: no such option $1. -h for help“;exit 1;*) break;esacdoneecho “opt_f is $opt_f“echo “opt_l is $opt_l“echo “first arg is $1“echo “2nd arg is $2“您可以这样运
21、行该脚本:cmdparser -l hello -f - -somefile1 somefile2返回的结果是:opt_f is 1opt_l is hellofirst arg is -somefile12nd arg is somefile2这个脚本是如何工作的呢?脚本首先在所有输入命令行参数中进行循环,将输入参数与 case 表达式进行比较,如果匹配则设置一个变量并且移除该参数。根据 unix 系统的惯例,首先输入的应该是包含减号的参数.第 2 部分 实例现在我们来讨论编写一个脚本的一般步骤。任何优秀的脚本都应该具有帮助和输入参数。并且写一个伪脚本(framework.sh),该脚本包含
22、了大多数脚本都需要的框架结构,是一个非常不错的主意。这时候,在写一个新的脚本时我们只需要执行一下 copy 命令:cp framework.sh myscript然后再插入自己的函数。让我们再看两个例子:二进制到十进制的转换脚本 b2d 将二进制数 (比如 1101) 转换为相应的十进制数。这也是一个用 expr 命令进行数学运算的例子:#!/bin/sh# vim: set sw=4 ts=4 et:help()cat = int2 |当 int1 大于/等于 int2 时, 返回 TrueInt1 int2 |当 int1 大于 int2 时,返回 TrueStr1 = str2 |当 s
23、tr1 与 str2 相同时, 返回 TrueStr1 != str2 |当 str1 与 str2 不同时, 返回 True-r file |当 file 是一个可读文件时, 返回 True-w file |当 file 是一个可写文件时, 返回 True-x file |当 file 是一个可执行文件时,返回 True-e file |当 file 存在时 ,返回 True-o file |当 file 文件的所有者是当前用户时, 返回 True-z file |当 file 长度为 0 时, 返回 True-f file |当 file 是一个普通文件时, 返回 True-d file
24、|当 file 是一个目录时 ,返回 TrueExp1 | exp2 |当 exp1 和 exp2 的值至少一个为 True 时, 返回 TrueExp1 当expression1 的条件为 false 并且 expression2 的条件满足为 True 时,shell 执行commands2 命令;当 expression1 和 expressin2 的条件值同为 false 时,shell 执行commands3 命令.if 语句以他的反写 fi 结尾.(2)case 语句case 语句要求 shell 将一个字符串 S 与一组字符串模式 P1,P2,.,Pn 比较, 当 S 与某个模式 Pi 想匹配时,就执行相应的那一部分程序 /命令.shell 的 case 语句中字符模式里可以包含象*这样的通配符.语法格式:bash/pdksh 用法:case string1 instr1)commands1;str2)commands2;*)commands3;esactcsh 用法:switch (string1)case str1:statements1breakswcase str2:statements2breakswdefault: