鸟哥的 Linux 私房菜
为取得较佳浏览结果,请爱用 firefox 浏览本网页
鸟哥的 Linux 私房菜馆 | 目录 | Linux 基础文件 | Linux 架站文件 | Linux 企业运用 | 安全管理 | 新手讨论 |
     
 
第十三章、学习 Shell Scripts
最近更新日期:2009/02/18
如果你真的很想要走资讯这条路,并且想要管理好属于你的主机,那么,别说鸟哥不告诉你, 可以自动管理系统的好工具: Shell scripts!这家伙真的是得要好好学习学习的! 基本上, shell script 有点像是早期的批次档,亦即是将一些指令汇整起来一次执行,但是 Shell script 拥有更强大的功能,那就是他可以进行类似程式 (program) 的撰写,并且不需要经过编译 (compile) 就能够执行, 真的很方便。加上我们可透过 shell script 来简化我们日常的工作管理, 而且,整个 Linux 环境中,一些服务 (services) 的启动都是透过 shell script 的, 如果你对于 script 不了解,嘿嘿!发生问题时,可真是会求助无门喔!所以,好好的学一学他吧!


大标题的图示什么是 Shell scripts

什么是 shell script (程式化脚本) 呢?就字面上的意义,我们将他分为两部份。 在‘ shell ’部分,我们在 十一章的 BASH 当中已经提过了,那是一个文字介面底下让我们与系统沟通的一个工具介面。那么‘ script ’是啥? 字面上的意义, script 是‘脚本、剧本’的意思。整句话是说, shell script 是针对 shell 所写的‘剧本!’

什么东西啊?其实, shell script 是利用 shell 的功能所写的一个‘程式 (program)’,这个程式是使用纯文字档,将一些 shell 的语法与指令(含外部指令)写在里面, 搭配正规表示法、管线命令与资料流重导向等功能,以达到我们所想要的处理目的。

所以,简单的说, shell script 就像是早期 DOS 年代的批次档 (.bat) ,最简单的功能就是将许多指令汇整写在一起, 让使用者很轻易的就能够 one touch 的方法去处理复杂的动作 (执行一个档案 "shell script" ,就能够一次执行多个指令)。 而且 shell script 更提供阵列、回圈、条件与逻辑判断等重要功能,让使用者也可以直接以 shell 来撰写程式,而不必使用类似 C 程式语言等传统程式撰写的语法呢!

这么说你可以了解了吗?是的! shell script 可以简单的被看成是批次档, 也可以被说成是一个程式语言,且这个程式语言由于都是利用 shell 与相关工具指令, 所以不需要编译即可执行,且拥有不错的除错 (debug) 工具,所以,他可以帮助系统管理员快速的管理好主机。


小标题的图示干嘛学习 shell scripts

这是个好问题:‘我又干嘛一定要学 shell script ?我又不是资讯人,没有写程式的概念, 那我干嘛还要学 shell script 呢?不要学可不可以啊?’呵呵~如果 Linux 对你而言, 你只是想要‘会用’而已,那么,不需要学 shell script 也还无所谓,这部分先给他跳过去, 等到有空的时候,再来好好的瞧一瞧。但是,如果你是真的想要玩清楚 Linux 的来龙去脉, 那么 shell script 就不可不知,为什么呢?因为:

  • 自动化管理的重要依据

    不用鸟哥说你也知道,管理一部主机真不是件简单的事情,每天要进行的任务就有: 查询登录档、追踪流量、监控使用者使用主机状态、主机各项硬体设备状态、 主机软体更新查询、更不要说得应付其他使用者的突然要求了。而这些工作的进行可以分为: (1)自行手动处理,或是 (2)写个简单的程式来帮你每日自动处理分析这两种方式,你觉得哪种方式比较好? 当然是让系统自动工作比较好,对吧!呵呵~这就得要良好的 shell script 来帮忙的啦!

  • 追踪与管理系统的重要工作

    虽然我们还没有提到服务启动的方法,不过,这里可以先提一下,我们 Linux 系统的服务 (services) 启动的介面是在 /etc/init.d/ 这个目录下,目录下的所有档案都是 scripts ; 另外,包括开机 (booting) 过程也都是利用 shell script 来帮忙搜寻系统的相关设定资料, 然后再代入各个服务的设定参数啊!举例来说,如果我们想要重新启动系统登录档, 可以使用:‘/etc/init.d/syslogd restart’,那个 syslogd 档案就是 script 啦!

    另外,鸟哥曾经在某一代的 Fedora 上面发现,启动 MySQL 这个资料库服务时,确实是可以启动的, 但是荧幕上却老是出现‘failure’!后来才发现,原来是启动 MySQL 那个 script 会主动的以‘空的密码’去尝试登入 MySQL ,但为了安全性鸟哥修改过 MySQL 的密码啰~当然就登入失败~ 后来改了改 script ,就略去这个问题啦!如此说来, script 确实是需要学习的啊!

  • 简单入侵侦测功能

    当我们的系统有异状时,大多会将这些异状记录在系统记录器,也就是我们常提到的‘系统登录档’, 那么我们可以在固定的几分钟内主动的去分析系统登录档,若察觉有问题,就立刻通报管理员, 或者是立刻加强防火墙的设定规则,如此一来,你的主机可就能够达到‘自我保护’的聪明学习功能啦~ 举例来说,我们可以通过 shell script 去分析‘当该封包尝试几次还是连线失败之后,就予以抵挡住该 IP’之类的举动,例如鸟哥写过一个关于抵挡砍站软体的 shell script , 就是用这个想法去达成的呢!

  • 连续指令单一化

    其实,对于新手而言, script 最简单的功能就是:‘汇整一些在 command line 下达的连续指令,将他写入 scripts 当中,而由直接执行 scripts 来启动一连串的 command line 指令输入!’例如: 防火墙连续规则 (iptables),开机载入程序的项目 (就是在 /etc/rc.d/rc.local 里头的资料) ,等等都是相似的功能啦! 其实,说穿了,如果不考虑 program 的部分,那么 scripts 也可以想成‘仅是帮我们把一大串的指令汇整在一个档案里面, 而直接执行该档案就可以执行那一串又臭又长的指令段!’就是这么简单啦!

  • 简易的资料处理

    由前一章正规表示法的 awk 程式说明中, 你可以发现, awk 可以用来处理简单的数据资料呢!例如薪资单的处理啊等等的。 shell script 的功能更强大,例如鸟哥曾经用 shell script 直接处理数据资料的比对啊, 文字资料的处理啊等等的,撰写方便,速度又快(因为在 Linux 效能较佳),真的是很不错用的啦!

  • 跨平台支援与学习历程较短

    几乎所有的 Unix Like 上面都可以跑 shell script ,连 MS Windows 系列也有相关的 script 模拟器可以用, 此外, shell script 的语法是相当亲和的,看都看的懂得文字 (虽然是英文),而不是机器码, 很容易学习~这些都是你可以加以考量的学习点啊!

上面这些都是你考虑学习 shell script 的特点~此外, shell script 还可以简单的以 vim 来直接编写,实在是很方便的好东西!所以,还是建议你学习一下啦。

不过,虽然 shell script 号称是程式 (program) ,但实际上, shell script 处理资料的速度上是不太够的。因为 shell script 用的是外部的指令与 bash shell 的一些预设工具,所以,他常常会去呼叫外部的函式库,因此,运算速度上面当然比不上传统的程式语言。 所以啰, shell script 用在系统管理上面是很好的一项工具,但是用在处理大量数值运算上, 就不够好了,因为 Shell scripts 的速度较慢,且使用的 CPU 资源较多,造成主机资源的分配不良。还好, 我们通常利用 shell script 来处理伺服器的侦测,倒是没有进行大量运算的需求啊!所以不必担心的啦!


小标题的图示第一支 script 的撰写与执行

如同前面讲到的,shell script 其实就是纯文字档,我们可以编辑这个档案,然后让这个档案来帮我们一次执行多个指令, 或者是利用一些运算与逻辑判断来帮我们达成某些功能。所以啦,要编辑这个档案的内容时,当然就需要具备有 bash 指令下达的相关认识。下达指令需要注意的事项在第五章的开始下达指令小节内已经提过,有疑问请自行回去翻阅。 在 shell script 的撰写中还需要用到底下的注意事项:

  1. 指令的执行是从上而下、从左而右的分析与执行;
  2. 指令的下达就如同第五章内提到的: 指令、选项与参数间的多个空白都会被忽略掉;
  3. 空白行也将被忽略掉,并且 [tab] 按键所推开的空白同样视为空白键;
  4. 如果读取到一个 Enter 符号 (CR) ,就尝试开始执行该行 (或该串) 命令;
  5. 至于如果一行的内容太多,则可以使用‘ \[Enter] ’来延伸至下一行;
  6. ‘ # ’可做为注解!任何加在 # 后面的资料将全部被视为注解文字而被忽略!

如此一来,我们在 script 内所撰写的程式,就会被一行一行的执行。现在我们假设你写的这个程式档名是 /home/dmtsai/shell.sh 好了,那如何执行这个档案?很简单,可以有底下几个方法:

  • 直接指令下达: shell.sh 档案必须要具备可读与可执行 (rx) 的权限,然后:
    • 绝对路径:使用 /home/dmtsai/shell.sh 来下达指令;
    • 相对路径:假设工作目录在 /home/dmtsai/ ,则使用 ./shell.sh 来执行
    • 变数‘PATH’功能:将 shell.sh 放在 PATH 指定的目录内,例如: ~/bin/

  • 以 bash 程式来执行:透过‘ bash shell.sh ’或‘ sh shell.sh ’来执行

反正重点就是要让那个 shell.sh 内的指令可以被执行的意思啦! 咦!那我为何需要使用 ‘./shell.sh ’来下达指令?忘记了吗?回去第十一章内的指令搜寻顺序察看一下, 你就会知道原因了!同时,由于 CentOS 预设使用者家目录下的 ~/bin 目录会被设定到 $PATH 内,所以你也可以将 shell.sh 建立在 /home/dmtsai/bin/ 底下 ( ~/bin 目录需要自行设定) 。此时,若 shell.sh 在 ~/bin 内且具有 rx 的权限,那就直接输入 shell.sh 即可执行该脚本程式

那为何‘ sh shell.sh ’也可以执行呢?这是因为 /bin/sh 其实就是 /bin/bash (连结档),使用 sh shell.sh 亦即告诉系统,我想要直接以 bash 的功能来执行 shell.sh 这个档案内的相关指令的意思,所以此时你的 shell.sh 只要有 r 的权限即可被执行喔!而我们也可以利用 sh 的参数,如 -n 及 -x 来检查与追踪 shell.sh 的语法是否正确呢! ^_^


  • 撰写第一支 script

在武侠世界中,不论是那个门派,要学武功要从扫地做起,那么要学程式呢?呵呵,肯定是由‘秀出 Hello World!’ 这个字眼开始的!OK!那么鸟哥就先写一支 script 给大家瞧一瞧:

[root@www ~]# mkdir scripts; cd scripts

[root@www scripts]# vi sh01.sh
#!/bin/bash
# Program:
#       This program shows "Hello World!" in your screen.
# History:
# 2005/08/23	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH
echo -e "Hello World! \a \n"
exit 0

在本章当中,请将所有撰写的 script 放置到你家目录的 ~/scripts 这个目录内, 未来比较好管理啦!上面的写法当中,鸟哥主要将整个程式的撰写分成数段,大致是这样:

  1. 第一行 #!/bin/bash 在宣告这个 script 使用的 shell 名称:
    因为我们使用的是 bash ,所以,必须要以‘ #!/bin/bash ’来宣告这个档案内的语法使用 bash 的语法!那么当这个程式被执行时,他就能够载入 bash 的相关环境设定档 (一般来说就是 non-login shell 的 ~/.bashrc), 并且执行 bash 来使我们底下的指令能够执行!这很重要的!(在很多状况中,如果没有设定好这一行, 那么该程式很可能会无法执行,因为系统可能无法判断该程式需要使用什么 shell 来执行啊!)

  2. 程式内容的说明:
    整个 script 当中,除了第一行的‘ #! ’是用来宣告 shell 的之外,其他的 # 都是‘注解’用途! 所以上面的程式当中,第二行以下就是用来说明整个程式的基本资料。一般来说, 建议你一定要养成说明该 script 的:1. 内容与功能; 2. 版本资讯; 3. 作者与联络方式; 4. 建档日期;5. 历史纪录 等等。这将有助于未来程式的改写与 debug 呢!

  3. 主要环境变数的宣告:
    建议务必要将一些重要的环境变数设定好,鸟哥个人认为, PATH 与 LANG (如果有使用到输出相关的资讯时) 是当中最重要的! 如此一来,则可让我们这支程式在进行时,可以直接下达一些外部指令,而不必写绝对路径呢!比较好啦!

  4. 主要程式部分
    就将主要的程式写好即可!在这个例子当中,就是 echo 那一行啦!

  5. 执行成果告知 (定义回传值)
    是否记得我们在第十一章里面要讨论一个指令的执行成功与否,可以使用 $? 这个变数来观察~ 那么我们也可以利用 exit 这个指令来让程式中断,并且回传一个数值给系统。 在我们这个例子当中,鸟哥使用 exit 0 ,这代表离开 script 并且回传一个 0 给系统, 所以我执行完这个 script 后,若接着下达 echo $? 则可得到 0 的值喔! 更聪明的读者应该也知道了,呵呵!利用这个 exit n (n 是数字) 的功能,我们还可以自订错误讯息, 让这支程式变得更加的 smart 呢!

接下来透过刚刚上头介绍的执行方法来执行看看结果吧!

[root@www scripts]# sh sh01.sh
Hello World !

你会看到荧幕是这样,而且应该还会听到‘咚’的一声,为什么呢?还记得前一章提到的 printf 吧?用 echo 接着那些特殊的按键也可以发生同样的事情~ 不过, echo 必须要加上 -e 的选项才行!呵呵!在你写完这个小 script 之后,你就可以大声的说:‘我也会写程式了’!哈哈! 很简单有趣吧~ ^_^

另外,你也可以利用:‘chmod a+x sh01.sh; ./sh01.sh’来执行这个 script 的呢!


小标题的图示撰写 shell script 的良好习惯建立

一个良好习惯的养成是很重要的~大家在刚开始撰写程式的时候,最容易忽略这部分, 认为程式写出来就好了,其他的不重要。其实,如果程式的说明能够更清楚,那么对你自己是有很大的帮助的。

举例来说,鸟哥自己为了自己的需求,曾经撰写了不少的 script 来帮我进行主机 IP 的侦测啊、 登录档分析与管理啊、自动上传下载重要设定档啊等等的,不过,早期就是因为太懒了, 管理的主机又太多了,常常同一个程式在不同的主机上面进行更改,到最后,到底哪一支才是最新的都记不起来, 而且,重点是,我到底是改了哪里?为什么做那样的修改?都忘的一干二净~真要命~

所以,后来鸟哥在写程式的时候,通常会比较仔细的将程式的设计过程给他记录下来,而且还会记录一些历史纪录, 如此一来,好多了~至少很容易知道我修改了哪些资料,以及程式修改的理念与逻辑概念等等, 在维护上面是轻松很多很多的喔!

另外,在一些环境的设定上面,毕竟每个人的环境都不相同,为了取得较佳的执行环境, 我都会自行先定义好一些一定会被用到的环境变数,例如 PATH 这个玩意儿! 这样比较好啦~所以说,建议你一定要养成良好的 script 撰写习惯,在每个 script 的档头处记录好:

  • script 的功能;
  • script 的版本资讯;
  • script 的作者与联络方式;
  • script 的版权宣告方式;
  • script 的 History (历史纪录);
  • script 内较特殊的指令,使用‘绝对路径’的方式来下达;
  • script 运作时需要的环境变数预先宣告与设定。

除了记录这些资讯之外,在较为特殊的程式码部分,个人建议务必要加上注解说明,可以帮助你非常非常多! 此外,程式码的撰写最好使用巢状方式,在包覆的内部程式码最好能以 [tab] 按键的空格向后推, 这样你的程式码会显的非常的漂亮与有条理!在查阅与 debug 上较为轻松愉快喔! 另外,使用撰写 script 的工具最好使用 vim 而不是 vi ,因为 vim 会有额外的语法检验机制,能够在第一阶段撰写时就发现语法方面的问题喔!


大标题的图示简单的 shell script 练习

在第一支 shell script 撰写完毕之后,相信你应该具有基本的撰写功力了。 接下来,在开始更深入的程式概念之前,我们先来玩一些简单的小范例好了。 底下的范例中,达成结果的方式相当的多,建议你先自行撰写看看,写完之后再与鸟哥写的内容比对, 这样才能更加深概念喔!好!不啰唆,我们就一个一个来玩吧!


小标题的图示简单范例

底下的范例在很多的脚本程式中都会用到,而底下的范例又都很简单!值得参考看看喔!


  • 对谈式脚本:变数内容由使用者决定

很多时候我们需要使用者输入一些内容,好让程式可以顺利运作。 简单的来说,大家应该都有安装过软体的经验,安装的时候,他不是会问你‘要安装到那个目录去’吗? 那个让使用者输入资料的动作,就是让使用者输入变数内容啦。

你应该还记得在十一章 bash 的时候,我们有学到一个 read 指令吧?现在,请你以 read 指令的用途,撰写一个 script ,他可以让使用者输入:1. first name 与 2. last name, 最后并且在荧幕上显示:‘Your full name is: ’的内容:

[root@www scripts]# vi sh02.sh
#!/bin/bash
# Program:
#	User inputs his first name and last name.  Program shows his full name.
# History:
# 2005/08/23	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

read -p "Please input your first name: " firstname  # 提示使用者输入
read -p "Please input your last name:  " lastname   # 提示使用者输入
echo -e "\nYour full name is: $firstname $lastname" # 结果由荧幕输出

将上面这个 sh02.sh 执行一下,你就能够发现使用者自己输入的变数可以让程式所取用,并且将他显示到荧幕上! 接下来,如果想要制作一个每次执行都会依据不同的日期而变化结果的脚本呢?


  • 随日期变化:利用 date 进行档案的建立

想像一个状况,假设我的伺服器内有资料库,资料库每天的资料都不太一样,因此当我备份时, 希望将每天的资料都备份成不同的档名,这样才能够让旧的资料也能够保存下来不被覆盖。 哇!不同档名呢!这真困扰啊?难道要我每天去修改 script ?

不需要啊!考虑每天的‘日期’并不相同,所以我可以将档名取成类似: backup.2009-02-14.data , 不就可以每天一个不同档名了吗?呵呵!确实如此。那个 2009-02-14 怎么来的?那就是重点啦!接下来出个相关的例子: 假设我想要建立三个空的档案 (透过 touch) ,档名最开头由使用者输入决定,假设使用者输入 filename 好了,那今天的日期是 2009/02/14 , 我想要以前天、昨天、今天的日期来建立这些档案,亦即 filename_20090212, filename_20090213, filename_20090214 ,该如何是好?

[root@www scripts]# vi sh03.sh
#!/bin/bash
# Program:
#	Program creates three files, which named by user's input 
#	and date command.
# History:
# 2005/08/23	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

# 1. 让使用者输入档案名称,并取得 fileuser 这个变数;
echo -e "I will use 'touch' command to create 3 files." # 纯粹显示资讯
read -p "Please input your filename: " fileuser         # 提示使用者输入

# 2. 为了避免使用者随意按 Enter ,利用变数功能分析档名是否有设定?

filename=${fileuser:-"filename"}           # 开始判断有否设定档名

# 3. 开始利用 date 指令来取得所需要的档名了;
date1=$(date --date='2 days ago' +%Y%m%d)  # 前两天的日期
date2=$(date --date='1 days ago' +%Y%m%d)  # 前一天的日期
date3=$(date +%Y%m%d)                      # 今天的日期
file1=${filename}${date1}                  # 底下三行在设定档名

file2=${filename}${date2}
file3=${filename}${date3}

# 4. 将档名建立吧!
touch "$file1"                             # 底下三行在建立档案
touch "$file2"
touch "$file3"

上面的范例鸟哥使用了很多在十一章介绍过的概念: 包括小指令‘ $(command) ’的取得讯息、变数的设定功能、变数的累加以及利用 touch 指令辅助! 如果你开始执行这个 sh03.sh 之后,你可以进行两次执行:一次直接按 [Enter] 来查阅档名是啥? 一次可以输入一些字元,这样可以判断你的脚本是否设计正确喔!


  • 数值运算:简单的加减乘除

各位看官应该还记得,我们可以使用 declare 来定义变数的类型吧? 当变数定义成为整数后才能够进行加减运算啊!此外,我们也可以利用‘ $((计算式)) ’来进行数值运算的。 可惜的是, bash shell 里头预设仅支援到整数的资料而已。OK!那我们来玩玩看,如果我们要使用者输入两个变数, 然后将两个变数的内容相乘,最后输出相乘的结果,那可以怎么做?

[root@www scripts]# vi sh04.sh
#!/bin/bash
# Program:
#	User inputs 2 integer numbers; program will cross these two numbers.
# History:
# 2005/08/23	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH
echo -e "You SHOULD input 2 numbers, I will cross them! \n"
read -p "first number:  " firstnu
read -p "second number: " secnu
total=$(($firstnu*$secnu))
echo -e "\nThe result of $firstnu x $secnu is ==> $total"

在数值的运算上,我们可以使用‘ declare -i total=$firstnu*$secnu ’ 也可以使用上面的方式来进行!基本上,鸟哥比较建议使用这样的方式来进行运算:

var=$((运算内容))

不但容易记忆,而且也比较方便的多,因为两个小括号内可以加上空白字元喔! 未来你可以使用这种方式来计算的呀!至于数值运算上的处理,则有:‘ +, -, *, /, % ’等等。 那个 % 是取余数啦~举例来说, 13 对 3 取余数,结果是 13=4*3+1,所以余数是 1 啊!就是:

[root@www scripts]# echo $(( 13 % 3 ))
1

这样了解了吧?多多学习与应用喔! ^_^


小标题的图示script 的执行方式差异 (source, sh script, ./script)

不同的 script 执行方式会造成不一样的结果喔!尤其影响 bash 的环境很大呢!脚本的执行方式除了前面小节谈到的方式之外,还可以利用 source 或小数点 (.) 来执行喔!那么这种执行方式有何不同呢?当然是不同的啦!让我们来说说!


  • 利用直接执行的方式来执行 script

当使用前一小节提到的直接指令下达 (不论是绝对路径/相对路径还是 $PATH 内),或者是利用 bash (或 sh) 来下达脚本时, 该 script 都会使用一个新的 bash 环境来执行脚本内的指令!也就是说,使用者种执行方式时, 其实 script 是在子程序的 bash 内执行的!我们在第十一章 BASH 内谈到 export 的功能时,曾经就父程序/子程序谈过一些概念性的问题, 重点在于:‘当子程序完成后,在子程序内的各项变数或动作将会结束而不会传回到父程序中’! 这是什么意思呢?

我们举刚刚提到过的 sh02.sh 这个脚本来说明好了,这个脚本可以让使用者自行设定两个变数,分别是 firstname 与 lastname,想一想,如果你直接执行该指令时,该指令帮你设定的 firstname 会不会生效?看一下底下的执行结果:

[root@www scripts]# echo $firstname $lastname
    <==确认了,这两个变数并不存在喔!
[root@www scripts]# sh sh02.sh
Please input your first name: VBird <==这个名字是鸟哥自己输入的

Please input your last name:  Tsai 

Your full name is: VBird Tsai      <==看吧!在 script 运作中,这两个变数有生效
[root@www scripts]# echo $firstname $lastname
    <==事实上,这两个变数在父程序的 bash 中还是不存在的!

上面的结果你应该会觉得很奇怪,怎么我已经利用 sh02.sh 设定好的变数竟然在 bash 环境底下无效!怎么回事呢? 如果将程序相关性绘制成图的话,我们以下图来说明。当你使用直接执行的方法来处理时,系统会给予一支新的 bash 让我们来执行 sh02.sh 里面的指令,因此你的 firstname, lastname 等变数其实是在下图中的子程序 bash 内执行的。 当 sh02.sh 执行完毕后,子程序 bash 内的所有资料便被移除,因此上表的练习中,在父程序底下 echo $firstname 时, 就看不到任何东西了!这样可以理解吗?

sh02.sh 在子程序中运作
图 2.2.1、sh02.sh 在子程序中运作


  • 利用 source 来执行脚本:在父程序中执行

如果你使用 source 来执行指令那就不一样了!同样的脚本我们来执行看看:

[root@www scripts]# source sh02.sh
Please input your first name: VBird
Please input your last name:  Tsai

Your full name is: VBird Tsai
[root@www scripts]# echo $firstname $lastname

VBird Tsai  <==嘿嘿!有资料产生喔!

竟然生效了!没错啊!因为 source 对 script 的执行方式可以使用底下的图示来说明! sh02.sh 会在父程序中执行的,因此各项动作都会在原本的 bash 内生效!这也是为啥你不登出系统而要让某些写入 ~/.bashrc 的设定生效时,需要使用‘ source ~/.bashrc ’而不能使用‘ bash ~/.bashrc ’是一样的啊!

sh02.sh 在父程序中运作
图 2.2.2、sh02.sh 在父程序中运作

大标题的图示善用判断式

在第十一章中,我们提到过 $? 这个变数所代表的意义, 此外,也透过 && 及 || 来作为前一个指令执行回传值对于后一个指令是否要进行的依据。第十一章的讨论中,如果想要判断一个目录是否存在, 当时我们使用的是 ls 这个指令搭配资料流重导向,最后配合 $? 来决定后续的指令进行与否。 但是否有更简单的方式可以来进行‘条件判断’呢?有的~那就是‘ test ’这个指令。


小标题的图示利用 test 指令的测试功能

当我要检测系统上面某些档案或者是相关的属性时,利用 test 这个指令来工作真是好用得不得了, 举例来说,我要检查 /dmtsai 是否存在时,使用:

[root@www ~]# test -e /dmtsai

执行结果并不会显示任何讯息,但最后我们可以透过 $? 或 && 及 || 来展现整个结果呢! 例如我们在将上面的例子改写成这样:

[root@www ~]# test -e /dmtsai && echo "exist" || echo "Not exist"
Not exist  <==结果显示不存在啊!

最终的结果可以告知我们是‘exist’还是‘Not exist’呢!那我知道 -e 是测试一个‘东西’在不在, 如果还想要测试一下该档名是啥玩意儿时,还有哪些标志可以来判断的呢?呵呵!有底下这些东西喔!

测试的标志代表意义
1. 关于某个档名的‘档案类型’判断,如 test -e filename 表示存在否
-e该‘档名’是否存在?(常用)
-f该‘档名’是否存在且为档案(file)?(常用)
-d该‘档名’是否存在且为目录(directory)?(常用)
-b该‘档名’是否存在且为一个 block device 装置?
-c该‘档名’是否存在且为一个 character device 装置?
-S该‘档名’是否存在且为一个 Socket 档案?
-p该‘档名’是否存在且为一个 FIFO (pipe) 档案?
-L该‘档名’是否存在且为一个连结档?
2. 关于档案的权限侦测,如 test -r filename 表示可读否 (但 root 权限常有例外)
-r侦测该档名是否存在且具有‘可读’的权限?
-w侦测该档名是否存在且具有‘可写’的权限?
-x侦测该档名是否存在且具有‘可执行’的权限?
-u侦测该档名是否存在且具有‘SUID’的属性?
-g侦测该档名是否存在且具有‘SGID’的属性?
-k侦测该档名是否存在且具有‘Sticky bit’的属性?
-s侦测该档名是否存在且为‘非空白档案’?
3. 两个档案之间的比较,如: test file1 -nt file2
-nt(newer than)判断 file1 是否比 file2 新
-ot(older than)判断 file1 是否比 file2 旧
-ef判断 file1 与 file2 是否为同一档案,可用在判断 hard link 的判定上。 主要意义在判定,两个档案是否均指向同一个 inode 哩!
4. 关于两个整数之间的判定,例如 test n1 -eq n2
-eq两数值相等 (equal)
-ne两数值不等 (not equal)
-gtn1 大于 n2 (greater than)
-ltn1 小于 n2 (less than)
-gen1 大于等于 n2 (greater than or equal)
-len1 小于等于 n2 (less than or equal)
5. 判定字串的资料
test -z string判定字串是否为 0 ?若 string 为空字串,则为 true
test -n string判定字串是否非为 0 ?若 string 为空字串,则为 false。
注: -n 亦可省略
test str1 = str2判定 str1 是否等于 str2 ,若相等,则回传 true
test str1 != str2判定 str1 是否不等于 str2 ,若相等,则回传 false
6. 多重条件判定,例如: test -r filename -a -x filename
-a(and)两状况同时成立!例如 test -r file -a -x file,则 file 同时具有 r 与 x 权限时,才回传 true。
-o(or)两状况任何一个成立!例如 test -r file -o -x file,则 file 具有 r 或 x 权限时,就可回传 true。
!反相状态,如 test ! -x file ,当 file 不具有 x 时,回传 true

OK!现在我们就利用 test 来帮我们写几个简单的例子。首先,判断一下,让使用者输入一个档名,我们判断:

  1. 这个档案是否存在,若不存在则给予一个‘Filename does not exist’的讯息,并中断程式;
  2. 若这个档案存在,则判断他是个档案或目录,结果输出‘Filename is regular file’或 ‘Filename is directory’
  3. 判断一下,执行者的身份对这个档案或目录所拥有的权限,并输出权限资料!

你可以先自行创作看看,然后再跟底下的结果讨论讨论。注意利用 test 与 && 还有 || 等标志!

[root@www scripts]# vi sh05.sh
#!/bin/bash
# Program:
#	User input a filename, program will check the flowing:
#	1.) exist? 2.) file/directory? 3.) file permissions 
# History:
# 2005/08/25	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

# 1. 让使用者输入档名,并且判断使用者是否真的有输入字串?

echo -e "Please input a filename, I will check the filename's type and \
permission. \n\n"
read -p "Input a filename : " filename
test -z $filename && echo "You MUST input a filename." && exit 0
# 2. 判断档案是否存在?若不存在则显示讯息并结束脚本
test ! -e $filename && echo "The filename '$filename' DO NOT exist" && exit 0
# 3. 开始判断档案类型与属性
test -f $filename && filetype="regulare file"
test -d $filename && filetype="directory"
test -r $filename && perm="readable"
test -w $filename && perm="$perm writable"
test -x $filename && perm="$perm executable"

# 4. 开始输出资讯!
echo "The filename: $filename is a $filetype"
echo "And the permissions are : $perm"

如果你执行这个脚本后,他会依据你输入的档名来进行检查喔!先看是否存在,再看为档案或目录类型,最后判断权限。 但是你必须要注意的是,由于 root 在很多权限的限制上面都是无效的,所以使用 root 执行这个脚本时, 常常会发现与 ls -l 观察到的结果并不相同!所以,建议使用一般使用者来执行这个脚本试看看。 不过你必须要使用 root 的身份先将这个脚本搬移给使用者就是了,不然一般使用者无法进入 /root 目录的。 很有趣的例子吧!你可以自行再以其他的案例来撰写一下可用的功能呢!


小标题的图示利用判断符号 [ ]

除了我们很喜欢使用的 test 之外,其实,我们还可以利用判断符号‘ [ ] ’(就是中括号啦) 来进行资料的判断呢! 举例来说,如果我想要知道 $HOME 这个变数是否为空的,可以这样做:

[root@www ~]# [ -z "$HOME" ] ; echo $?

使用中括号必须要特别注意,因为中括号用在很多地方,包括万用字元与正规表示法等等,所以如果要在 bash 的语法当中使用中括号作为 shell 的判断式时,必须要注意中括号的两端需要有空白字元来分隔喔! 假设我空白键使用‘□’符号来表示,那么,在这些地方你都需要有空白键:

[  "$HOME"  ==  "$MAIL"  ]
[□"$HOME"□==□"$MAIL"□]
 ↑       ↑  ↑       ↑
Tips:
你会发现鸟哥在上面的判断式当中使用了两个等号‘ == ’。其实在 bash 当中使用一个等号与两个等号的结果是一样的! 不过在一般惯用程式的写法中,一个等号代表‘变数的设定’,两个等号则是代表‘逻辑判断 (是否之意)’。 由于我们在中括号内重点在于‘判断’而非‘设定变数’,因此鸟哥建议您还是使用两个等号较佳!
鸟哥的图示

上面的例子在说明,两个字串 $HOME 与 $MAIL 是否相同的意思,相当于 test $HOME = $MAIL 的意思啦! 而如果没有空白分隔,例如 [$HOME==$MAIL] 时,我们的 bash 就会显示错误讯息了!这可要很注意啊! 所以说,你最好要注意:

  • 在中括号 [] 内的每个元件都需要有空白键来分隔;
  • 在中括号内的变数,最好都以双引号括号起来;
  • 在中括号内的常数,最好都以单或双引号括号起来。

为什么要这么麻烦啊?直接举例来说,假如我设定了 name="VBird Tsai" ,然后这样判定:

[root@www ~]# name="VBird Tsai"

[root@www ~]# [ $name == "VBird" ]
bash: [: too many arguments

见鬼了!怎么会发生错误啊?bash 还跟我说错误是由于‘太多参数 (arguments)’所致! 为什么呢?因为 $name 如果没有使用双引号刮起来,那么上面的判定式会变成:

[ VBird Tsai == "VBird" ]

上面肯定不对嘛!因为一个判断式仅能有两个资料的比对,上面 VBird 与 Tsai 还有 "VBird" 就有三个资料! 这不是我们要的!我们要的应该是底下这个样子:

[ "VBird Tsai" == "VBird" ]

这可是差很多的喔!另外,中括号的使用方法与 test 几乎一模一样啊~ 只是中括号比较常用在条件判断式 if ..... then ..... fi 的情况中就是了。 好,那我们也使用中括号的判断来做一个小案例好了,案例设定如下:

  1. 当执行一个程式的时候,这个程式会让使用者选择 Y 或 N ,
  2. 如果使用者输入 Y 或 y 时,就显示‘ OK, continue ’
  3. 如果使用者输入 n 或 N 时,就显示‘ Oh, interrupt !’
  4. 如果不是 Y/y/N/n 之内的其他字元,就显示‘ I don't know what your choice is ’
利用中括号、 && 与 || 来继续吧!
[root@www scripts]# vi sh06.sh

#!/bin/bash
# Program:
# 	This program shows the user's choice
# History:
# 2005/08/25	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

read -p "Please input (Y/N): " yn
[ "$yn" == "Y" -o "$yn" == "y" ] && echo "OK, continue" && exit 0
[ "$yn" == "N" -o "$yn" == "n" ] && echo "Oh, interrupt!" && exit 0
echo "I don't know what your choice is" && exit 0

由于输入正确 (Yes) 的方法有大小写之分,不论输入大写 Y 或小写 y 都是可以的,此时判断式内就得要有两个判断才行! 由于是任何一个成立即可 (大小或小写的 y) ,所以这里使用 -o (或) 连结两个判断喔! 很有趣吧!利用这个字串判别的方法,我们就可以很轻松的将使用者想要进行的工作分门别类呢! 接下来,我们再来谈一些其他有的没有的东西吧!


小标题的图示Shell script 的预设变数($0, $1...)

我们知道指令可以带有选项与参数,例如 ls -la 可以察看包含隐藏档的所有属性与权限。那么 shell script 能不能在脚本档名后面带有参数呢?很有趣喔!举例来说,如果你想要重新启动系统登录档的功能,可以这样做:

[root@www ~]# file /etc/init.d/syslog
/etc/init.d/syslog: Bourne-Again shell script text executable

# 使用 file 来查询后,系统告知这个档案是个 bash 的可执行 script 喔!
[root@www ~]# /etc/init.d/syslog restart

restart 是除新启动的意思,上面的指令可以‘重新启动 /etc/init.d/syslog 这支程式’的意思! 唔!那么如果你在 /etc/init.d/syslog 后面加上 stop 呢?没错!就可以直接关闭该服务了!这么神奇啊? 没错啊!如果你要依据程式的执行给予一些变数去进行不同的任务时,本章一开始是使用 read 的功能!但 read 功能的问题是你得要手动由键盘输入一些判断式。如果透过指令后面接参数, 那么一个指令就能够处理完毕而不需要手动再次输入一些变数行为!这样下达指令会比较简单方便啦!

script 是怎么达成这个功能的呢?其实 script 针对参数已经有设定好一些变数名称了!对应如下:

/path/to/scriptname  opt1  opt2  opt3  opt4 
       $0             $1    $2    $3    $4

这样够清楚了吧?执行的脚本档名为 $0 这个变数,第一个接的参数就是 $1 啊~ 所以,只要我们在 script 里面善用 $1 的话,就可以很简单的立即下达某些指令功能了!除了这些数字的变数之外, 我们还有一些较为特殊的变数可以在 script 内使用来呼叫这些参数喔!

  • $# :代表后接的参数‘个数’,以上表为例这里显示为‘ 4 ’;
  • $@ :代表‘ "$1" "$2" "$3" "$4" ’之意,每个变数是独立的(用双引号括起来);
  • $* :代表‘ "$1c$2c$3c$4" ’,其中 c 为分隔字元,预设为空白键, 所以本例中代表‘ "$1 $2 $3 $4" ’之意。

那个 $@ 与 $* 基本上还是有所不同啦!不过,一般使用情况下可以直接记忆 $@ 即可! 好了,来做个例子吧~假设我要执行一个可以携带参数的 script ,执行该脚本后荧幕会显示如下的资料:

  • 程式的档名为何?
  • 共有几个参数?
  • 若参数的个数小于 2 则告知使用者参数数量太少
  • 全部的参数内容为何?
  • 第一个参数为何?
  • 第二个参数为何
[root@www scripts]# vi sh07.sh
#!/bin/bash
# Program:
#	Program shows the script name, parameters...
# History:
# 2009/02/17	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

echo "The script name is        ==> $0"
echo "Total parameter number is ==> $#"
[ "$#" -lt 2 ] && echo "The number of parameter is less than 2.  Stop here." \
	&& exit 0
echo "Your whole parameter is   ==> '$@'"
echo "The 1st parameter         ==> $1"
echo "The 2nd parameter         ==> $2"

执行结果如下:

[root@www scripts]# sh sh07.sh theone haha quot
The script name is        ==> sh07.sh            <==档名
Total parameter number is ==> 3                  <==果然有三个参数

Your whole parameter is   ==> 'theone haha quot' <==参数的内容全部
The 1st parameter         ==> theone             <==第一个参数
The 2nd parameter         ==> haha               <==第二个参数


  • shift:造成参数变数号码偏移

除此之外,脚本后面所接的变数是否能够进行偏移 (shift) 呢?什么是偏移啊?我们直接以底下的范例来说明好了, 用范例说明比较好解释!我们将 sh07.sh 的内容稍作变化一下,用来显示每次偏移后参数的变化情况:

[root@www scripts]# vi sh08.sh
#!/bin/bash
# Program:
#	Program shows the effect of shift function.
# History:
# 2009/02/17	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

echo "Total parameter number is ==> $#"
echo "Your whole parameter is   ==> '$@'"
shift   # 进行第一次‘一个变数的 shift ’

echo "Total parameter number is ==> $#"
echo "Your whole parameter is   ==> '$@'"
shift 3 # 进行第二次‘三个变数的 shift ’
echo "Total parameter number is ==> $#"
echo "Your whole parameter is   ==> '$@'"

这玩意的执行成果如下:

[root@www scripts]# sh sh08.sh one two three four five six <==给予六个参数
Total parameter number is ==> 6   <==最原始的参数变数情况
Your whole parameter is   ==> 'one two three four five six'
Total parameter number is ==> 5   <==第一次偏移,看底下发现第一个 one 不见了

Your whole parameter is   ==> 'two three four five six'
Total parameter number is ==> 2   <==第二次偏移掉三个,two three four 不见了
Your whole parameter is   ==> 'five six'

光看结果你就可以知道啦,那个 shift 会移动变数,而且 shift 后面可以接数字,代表拿掉最前面的几个参数的意思。 上面的执行结果中,第一次进行 shift 后他的显示情况是‘ one two three four five six’,所以就剩下五个啦!第二次直接拿掉三个,就变成‘ two three four five six ’啦! 这样这个案例可以了解了吗?理解了 shift 的功能了吗?

上面这8个例子都很简单吧?几乎都是利用 bash 的相关功能而已~ 不难啦~底下我们就要使用条件判断式来进行一些分别功能的设定了,好好瞧一瞧先~


大标题的图示条件判断式

只要讲到‘程式’的话,那么条件判断式,亦即是‘ if then ’这种判别式肯定一定要学习的! 因为很多时候,我们都必须要依据某些资料来判断程式该如何进行。举例来说,我们在上头的 sh06.sh 范例中不是有练习当使用者输入 Y/N 时,必须要执行不同的讯息输出吗?简单的方式可以利用 && 与 || ,但如果我还想要执行一堆指令呢?那真的得要 if then 来帮忙啰~底下我们就来聊一聊!


小标题的图示利用 if .... then

这个 if .... then 是最常见的条件判断式了~简单的说,就是当符合某个条件判断的时候, 就予以进行某项工作就是了。这个 if ... then 的判断还有多层次的情况!我们分别介绍如下:


  • 单层、简单条件判断式

如果你只有一个判断式要进行,那么我们可以简单的这样看:

if [ 条件判断式 ]; then
	当条件判断式成立时,可以进行的指令工作内容;
fi   <==将 if 反过来写,就成为 fi 啦!结束 if 之意!

至于条件判断式的判断方法,与前一小节的介绍相同啊!较特别的是,如果我有多个条件要判别时, 除了 sh06.sh 那个案例所写的,也就是‘将多个条件写入一个中括号内的情况’之外, 我还可以有多个中括号来隔开喔!而括号与括号之间,则以 && 或 || 来隔开,他们的意义是:

  • && 代表 AND ;
  • || 代表 or ;

所以,在使用中括号的判断式中, && 及 || 就与指令下达的状态不同了。举例来说, sh06.sh 里面的判断式可以这样修改:

[ "$yn" == "Y" -o "$yn" == "y" ]
上式可替换为
[ "$yn" == "Y" ] || [ "$yn" == "y" ]

之所以这样改,很多人是习惯问题!很多人则是喜欢一个中括号仅有一个判别式的原因。好了, 现在我们来将 sh06.sh 这个脚本修改成为 if ... then 的样式来看看:

[root@www scripts]# cp sh06.sh sh06-2.sh  <==用改的比较快!

[root@www scripts]# vi sh06-2.sh
#!/bin/bash
# Program:
#       This program shows the user's choice
# History:
# 2005/08/25    VBird   First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

read -p "Please input (Y/N): " yn

if [ "$yn" == "Y" ] || [ "$yn" == "y" ]; then
	echo "OK, continue"
	exit 0
fi
if [ "$yn" == "N" ] || [ "$yn" == "n" ]; then
	echo "Oh, interrupt!"
	exit 0
fi
echo "I don't know what your choice is" && exit 0

不过,由这个例子看起来,似乎也没有什么了不起吧? sh06.sh 还比较简单呢~ 但是如果以逻辑概念来看,其实上面的范例中,我们使用了两个条件判断呢!明明仅有一个 $yn 的变数,为何需要进行两次比对呢? 此时,多重条件判断就能够来测试测试啰!


  • 多重、复杂条件判断式

在同一个资料的判断中,如果该资料需要进行多种不同的判断时,应该怎么作?举例来说,上面的 sh06.sh 脚本中,我们只要进行一次 $yn 的判断就好 (仅进行一次 if ),不想要作多次 if 的判断。 此时你就得要知道底下的语法了:

# 一个条件判断,分成功进行与失败进行 (else)
if [ 条件判断式 ]; then
	当条件判断式成立时,可以进行的指令工作内容;

else
	当条件判断式不成立时,可以进行的指令工作内容;
fi

如果考虑更复杂的情况,则可以使用这个语法:

# 多个条件判断 (if ... elif ... elif ... else) 分多种不同情况执行
if [ 条件判断式一 ]; then

	当条件判断式一成立时,可以进行的指令工作内容;
elif [ 条件判断式二 ]; then
	当条件判断式二成立时,可以进行的指令工作内容;
else
	当条件判断式一与二均不成立时,可以进行的指令工作内容;
fi

你得要注意的是, elif 也是个判断式,因此出现 elif 后面都要接 then 来处理!但是 else 已经是最后的没有成立的结果了, 所以 else 后面并没有 then 喔!好!我们来将 sh06-2.sh 改写成这样:

[root@www scripts]# cp sh06-2.sh sh06-3.sh
[root@www scripts]# vi sh06-3.sh
#!/bin/bash
# Program:
#       This program shows the user's choice
# History:
# 2005/08/25    VBird   First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

read -p "Please input (Y/N): " yn

if [ "$yn" == "Y" ] || [ "$yn" == "y" ]; then
	echo "OK, continue"
elif [ "$yn" == "N" ] || [ "$yn" == "n" ]; then
	echo "Oh, interrupt!"
else
	echo "I don't know what your choice is"
fi

是否程式变得很简单,而且依序判断,可以避免掉重复判断的状况,这样真的很容易设计程式的啦! ^_^! 好了,让我们再来进行另外一个案例的设计。一般来说,如果你不希望使用者由键盘输入额外的资料时, 可以使用上一节提到的参数功能 ($1)!让使用者在下达指令时就将参数带进去! 现在我们想让使用者输入‘ hello ’这个关键字时,利用参数的方法可以这样依序设计:

  1. 判断 $1 是否为 hello,如果是的话,就显示 "Hello, how are you ?";
  2. 如果没有加任何参数,就提示使用者必须要使用的参数下达法;
  3. 而如果加入的参数不是 hello ,就提醒使用者仅能使用 hello 为参数。

整个程式的撰写可以是这样的:

[root@www scripts]# vi sh09.sh
#!/bin/bash
# Program:
#	Check $1 is equal to "hello"
# History:
# 2005/08/28	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

if [ "$1" == "hello" ]; then
	echo "Hello, how are you ?"
elif [ "$1" == "" ]; then
	echo "You MUST input parameters, ex> {$0 someword}"
else
	echo "The only parameter is 'hello', ex> {$0 hello}"
fi

然后你可以执行这支程式,分别在 $1 的位置输入 hello, 没有输入与随意输入, 就可以看到不同的输出啰~是否还觉得挺简单的啊! ^_^。事实上, 学到这里,也真的很厉害了~好了,底下我们继续来玩一些比较大一点的计划啰~

我们在第十一章已经学会了 grep 这个好用的玩意儿,那么多学一个叫做 netstat 的指令,这个指令可以查询到目前主机有开启的网路服务埠口 (service ports), 相关的功能我们会在伺服器架设篇继续介绍,这里你只要知道,我可以利用‘ netstat -tuln ’来取得目前主机有启动的服务, 而且取得的资讯有点像这样:


[root@www ~]# netstat -tuln
Active Internet connections (only servers)
Proto Recv-Q Send-Q Local Address     Foreign Address   State
tcp        0      0 0.0.0.0:111       0.0.0.0:*         LISTEN
tcp        0      0 127.0.0.1:631     0.0.0.0:*         LISTEN
tcp        0      0 127.0.0.1:25      0.0.0.0:*         LISTEN
tcp        0      0 :::22             :::*              LISTEN
udp        0      0 0.0.0.0:111       0.0.0.0:*
udp        0      0 0.0.0.0:631       0.0.0.0:*
#封包格式           本地IP:埠口       远端IP:埠口       是否监听

上面的重点是‘Local Address (本地主机的IP与埠口对应)’那个栏位,他代表的是本机所启动的网路服务! IP的部分说明的是该服务位于那个介面上,若为 127.0.0.1 则是仅针对本机开放,若是 0.0.0.0 或 ::: 则代表对整个 Internet 开放 (更多资讯请参考伺服器架设篇的介绍)。 每个埠口 (port) 都有其特定的网路服务,几个常见的 port 与相关网路服务的关系是:

  • 80: WWW
  • 22: ssh
  • 21: ftp
  • 25: mail
  • 111: RPC(远端程序呼叫)
  • 631: CUPS(列印服务功能)

假设我的主机有兴趣要侦测的是比较常见的 port 21, 22, 25及 80 时,那我如何透过 netstat 去侦测我的主机是否有开启这四个主要的网路服务埠口呢?由于每个服务的关键字都是接在冒号‘ : ’后面, 所以可以藉由撷取类似‘ :80 ’来侦测的!那我就可以简单的这样去写这个程式喔:

[root@www scripts]# vi sh10.sh
#!/bin/bash
# Program:
# 	Using netstat and grep to detect WWW,SSH,FTP and Mail services.
# History:
# 2005/08/28	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

# 1. 先作一些告知的动作而已~
echo "Now, I will detect your Linux server's services!"
echo -e "The www, ftp, ssh, and mail will be detect! \n"

# 2. 开始进行一些测试的工作,并且也输出一些资讯啰!
testing=$(netstat -tuln | grep ":80 ")   # 侦测看 port 80 在否?

if [ "$testing" != "" ]; then
	echo "WWW is running in your system."
fi
testing=$(netstat -tuln | grep ":22 ")   # 侦测看 port 22 在否?
if [ "$testing" != "" ]; then
	echo "SSH is running in your system."
fi
testing=$(netstat -tuln | grep ":21 ")   # 侦测看 port 21 在否?
if [ "$testing" != "" ]; then
	echo "FTP is running in your system."
fi
testing=$(netstat -tuln | grep ":25 ")   # 侦测看 port 25 在否?
if [ "$testing" != "" ]; then
	echo "Mail is running in your system."
fi

实际执行这支程式你就可以看到你的主机有没有启动这些服务啦!是否很有趣呢? 条件判断式还可以搞的更复杂!举例来说,在台湾当兵是国民应尽的义务,不过,在当兵的时候总是很想要退伍的! 那你能不能写个脚本程式来跑,让使用者输入他的退伍日期,让你去帮他计算还有几天才退伍?

由于日期是要用相减的方式来处置,所以我们可以透过使用 date 显示日期与时间,将他转为由 1970-01-01 累积而来的秒数, 透过秒数相减来取得剩余的秒数后,再换算为日数即可。整个脚本的制作流程有点像这样:

  1. 先让使用者输入他们的退伍日期;
  2. 再由现在日期比对退伍日期;
  3. 由两个日期的比较来显示‘还需要几天’才能够退伍的字样。

似乎挺难的样子?其实也不会啦,利用‘ date --date="YYYYMMDD" +%s ’转成秒数后,接下来的动作就容易的多了!如果你已经写完了程式,对照底下的写法试看看:

[root@www scripts]# vi sh11.sh
#!/bin/bash
# Program:
#	You input your demobilization date, I calculate how many days
#	before you demobilize.
# History:
# 2005/08/29	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

# 1. 告知使用者这支程式的用途,并且告知应该如何输入日期格式?
echo "This program will try to calculate :"
echo "How many days before your demobilization date..."
read -p "Please input your demobilization date (YYYYMMDD ex>20090401): " date2


# 2. 测试一下,这个输入的内容是否正确?利用正规表示法啰~
date_d=$(echo $date2 |grep '[0-9]\{8\}')   # 看看是否有八个数字
if [ "$date_d" == "" ]; then
	echo "You input the wrong date format...."
	exit 1
fi

# 3. 开始计算日期啰~
declare -i date_dem=`date --date="$date2" +%s`    # 退伍日期秒数
declare -i date_now=`date +%s`                    # 现在日期秒数
declare -i date_total_s=$(($date_dem-$date_now))  # 剩余秒数统计

declare -i date_d=$(($date_total_s/60/60/24))     # 转为日数
if [ "$date_total_s" -lt "0" ]; then              # 判断是否已退伍
	echo "You had been demobilization before: " $((-1*$date_d)) " ago"
else
	declare -i date_h=$(($(($date_total_s-$date_d*60*60*24))/60/60))
	echo "You will demobilize after $date_d days and $date_h hours."
fi

瞧一瞧,这支程式可以帮你计算退伍日期呢~如果是已经退伍的朋友, 还可以知道已经退伍多久了~哈哈!很可爱吧~脚本中的 date_d 变数宣告那个 /60/60/24 是来自于一天的总秒数 (24小时*60分*60秒) 。瞧~全部的动作都没有超出我们所学的范围吧~ ^_^ 还能够避免使用者输入错误的数字,所以多了一个正规表示法的判断式呢~ 这个例子比较难,有兴趣想要一探究竟的朋友,可以作一下课后练习题 关于计算生日的那一题喔!~加油!


小标题的图示利用 case ..... esac 判断

上个小节提到的‘ if .... then .... fi ’对于变数的判断是以‘比对’的方式来分辨的, 如果符合状态就进行某些行为,并且透过较多层次 (就是 elif ...) 的方式来进行多个变数的程式码撰写,譬如 sh09.sh 那个小程式,就是用这样的方式来撰写的啰。 好,那么万一我有多个既定的变数内容,例如 sh09.sh 当中,我所需要的变数就是 "hello" 及空字串两个, 那么我只要针对这两个变数来设定状况就好了,对吧?那么可以使用什么方式来设计呢?呵呵~就用 case ... in .... esac 吧~,他的语法如下:

case  $变数名称 in   <==关键字为 case ,还有变数前有钱字号

  "第一个变数内容")   <==每个变数内容建议用双引号括起来,关键字则为小括号 )
	程式段
	;;            <==每个类别结尾使用两个连续的分号来处理!
  "第二个变数内容")
	程式段
	;;

  *)                  <==最后一个变数内容都会用 * 来代表所有其他值
	不包含第一个变数内容与第二个变数内容的其他程式执行段
	exit 1
	;;
esac                  <==最终的 case 结尾!‘反过来写’思考一下!

要注意的是,这个语法以 case (实际案例之意) 为开头,结尾自然就是将 case 的英文反过来写!就成为 esac 啰! 不会很难背啦!另外,每一个变数内容的程式段最后都需要两个分号 (;;) 来代表该程式段落的结束,这挺重要的喔! 至于为何需要有 * 这个变数内容在最后呢?这是因为,如果使用者不是输入变数内容一或二时, 我们可以告知使用者相关的资讯啊!废话少说,我们拿 sh09.sh 的案例来修改一下,他应该会变成这样喔:

[root@www scripts]# vi sh09-2.sh
#!/bin/bash
# Program:
# 	Show "Hello" from $1.... by using case .... esac
# History:
# 2005/08/29	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

case $1 in
  "hello")
	echo "Hello, how are you ?"
	;;
  "")
	echo "You MUST input parameters, ex> {$0 someword}"
	;;
  *)   # 其实就相当于万用字元,0~无穷多个任意字元之意!
	echo "Usage $0 {hello}"
	;;
esac

在上面这个 sh09-2.sh 的案例当中,如果你输入‘ sh sh09-2.sh test ’来执行, 那么荧幕上就会出现‘Usage sh09-2.sh {hello}’的字样,告知执行者仅能够使用 hello 喔~ 这样的方式对于需要某些固定字串来执行的变数内容就显的更加的方便呢! 这种方式你真的要熟悉喔!这是因为系统的很多服务的启动 scripts 都是使用这种写法的, 举例来说,我们 Linux 的服务启动放置目录是在 /etc/init.d/ 当中,我已经知道里头有个 syslog 的服务,我想要重新启动这个服务,可以这样做:

/etc/init.d/syslog restart

重点是那个 restart 啦!如果你使用‘ less /etc/init.d/syslog ’去查阅一下,就会看到他使用的是 case 语法, 并且会规定某些既定的变数内容,你可以直接下达 /etc/init.d/syslog , 该 script 就会告知你有哪些后续接的变数可以使用啰~方便吧! ^_^

一般来说,使用‘ case $变数 in ’这个语法中,当中的那个‘ $变数 ’大致有两种取得的方式:

  • 直接下达式:例如上面提到的,利用‘ script.sh variable ’ 的方式来直接给予 $1 这个变数的内容,这也是在 /etc/init.d 目录下大多数程式的设计方式。

  • 互动式:透过 read 这个指令来让使用者输入变数的内容。

这么说或许你的感受性还不高,好,我们直接写个程式来玩玩:让使用者能够输入 one, two, three , 并且将使用者的变数显示到荧幕上,如果不是 one, two, three 时,就告知使用者仅有这三种选择。

[root@www scripts]# vi sh12.sh
#!/bin/bash
# Program:
#	This script only accepts the flowing parameter: one, two or three.
# History:
# 2005/08/29	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

echo "This program will print your selection !"
# read -p "Input your choice: " choice # 暂时取消,可以替换!
# case $choice in                      # 暂时取消,可以替换!

case $1 in                             # 现在使用,可以用上面两行替换!
  "one")
	echo "Your choice is ONE"
	;;
  "two")
	echo "Your choice is TWO"
	;;
  "three")
	echo "Your choice is THREE"
	;;
  *)
	echo "Usage $0 {one|two|three}"
	;;
esac

此时,你可以使用‘ sh sh12.sh two ’的方式来下达指令,就可以收到相对应的回应了。 上面使用的是直接下达的方式,而如果使用的是互动式时,那么将上面第 10, 11 行的 "#" 拿掉, 并将 12 行加上注解 (#),就可以让使用者输入参数啰~这样是否很有趣啊?


小标题的图示利用 function 功能

什么是‘函数 (function)’功能啊?简单的说,其实, 函数可以在 shell script 当中做出一个类似自订执行指令的东西,最大的功能是, 可以简化我们很多的程式码~举例来说,上面的 sh12.sh 当中,每个输入结果 one, two, three 其实输出的内容都一样啊~那么我就可以使用 function 来简化了! function 的语法是这样的:

function fname() {
	程式段
}

那个 fname 就是我们的自订的执行指令名称~而程式段就是我们要他执行的内容了。 要注意的是,因为 shell script 的执行方式是由上而下,由左而右, 因此在 shell script 当中的 function 的设定一定要在程式的最前面, 这样才能够在执行时被找到可用的程式段喔!好~我们将 sh12.sh 改写一下,自订一个名为 printit 的函数来使用喔:

[root@www scripts]# vi sh12-2.sh
#!/bin/bash
# Program:
#	Use function to repeat information.
# History:
# 2005/08/29	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

function printit(){
	echo -n "Your choice is "     # 加上 -n 可以不断行继续在同一行显示
}

echo "This program will print your selection !"
case $1 in
  "one")
	printit; echo $1 | tr 'a-z' 'A-Z'  # 将参数做大小写转换!

	;;
  "two")
	printit; echo $1 | tr 'a-z' 'A-Z'
	;;
  "three")
	printit; echo $1 | tr 'a-z' 'A-Z'
	;;
  *)
	echo "Usage $0 {one|two|three}"
	;;
esac

以上面的例子来说,鸟哥做了一个函数名称为 printit ,所以,当我在后续的程式段里面, 只要执行 printit 的话,就表示我的 shell script 要去执行‘ function printit .... ’ 里面的那几个程式段落啰!当然啰,上面这个例子举得太简单了,所以你不会觉得 function 有什么好厉害的, 不过,如果某些程式码一再地在 script 当中重复时,这个 function 可就重要的多啰~ 不但可以简化程式码,而且可以做成类似‘模组’的玩意儿,真的很棒啦!

Tips:
建议读者可以使用类似 vim 的编辑器到 /etc/init.d/ 目录下去查阅一下你所看到的档案, 并且自行追踪一下每个档案的执行情况,相信会更有心得!
鸟哥的图示

另外, function 也是拥有内建变数的~他的内建变数与 shell script 很类似, 函数名称代表示 $0 ,而后续接的变数也是以 $1, $2... 来取代的~ 这里很容易搞错喔~因为‘ function fname() { 程式段 } ’内的 $0, $1... 等等与 shell script 的 $0 是不同的。以上面 sh12-2.sh 来说,假如我下达:‘ sh sh12-2.sh one ’ 这表示在 shell script 内的 $1 为 "one" 这个字串。但是在 printit() 内的 $1 则与这个 one 无关。 我们将上面的例子再次的改写一下,让你更清楚!

[root@www scripts]# vi sh12-3.sh
#!/bin/bash
# Program:
#	Use function to repeat information.
# History:
# 2005/08/29	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

function printit(){
	echo "Your choice is $1"   # 这个 $1 必须要参考底下指令的下达
}

echo "This program will print your selection !"
case $1 in
  "one")
	printit 1  # 请注意, printit 指令后面还有接参数!

	;;
  "two")
	printit 2
	;;
  "three")
	printit 3
	;;
  *)
	echo "Usage $0 {one|two|three}"
	;;
esac

在上面的例子当中,如果你输入‘ sh sh12-3.sh one ’就会出现‘ Your choice is 1 ’的字样~ 为什么是 1 呢?因为在程式段落当中,我们是写了‘ printit 1 ’那个 1 就会成为 function 当中的 $1 喔~ 这样是否理解呢? function 本身其实比较困难一点,如果你还想要进行其他的撰写的话。 不过,我们仅是想要更加了解 shell script 而已,所以,这里看看即可~了解原理就好啰~ ^_^


大标题的图示回圈 (loop)

除了 if...then...fi 这种条件判断式之外,回圈可能是程式当中最重要的一环了~ 回圈可以不断的执行某个程式段落,直到使用者设定的条件达成为止。 所以,重点是那个‘条件的达成’是什么。除了这种依据判断式达成与否的不定回圈之外, 还有另外一种已经固定要跑多少次的回圈形态,可称为固定回圈的形态呢!底下我们就来谈一谈:


小标题的图示while do done, until do done (不定回圈)

一般来说,不定回圈最常见的就是底下这两种状态了:

while [ condition ]  <==中括号内的状态就是判断式
do            <==do 是回圈的开始!

	程式段落
done          <==done 是回圈的结束

while 的中文是‘当....时’,所以,这种方式说的是‘当 condition 条件成立时,就进行回圈,直到 condition 的条件不成立才停止’的意思。还有另外一种不定回圈的方式:

until [ condition ]
do

	程式段落
done

这种方式恰恰与 while 相反,它说的是‘当 condition 条件成立时,就终止回圈, 否则就持续进行回圈的程式段。’是否刚好相反啊~我们以 while 来做个简单的练习好了。 假设我要让使用者输入 yes 或者是 YES 才结束程式的执行,否则就一直进行告知使用者输入字串。

[root@www scripts]# vi sh13.sh
#!/bin/bash
# Program:
#	Repeat question until user input correct answer.
# History:
# 2005/08/29	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

while [ "$yn" != "yes" -a "$yn" != "YES" ]
do
	read -p "Please input yes/YES to stop this program: " yn
done
echo "OK! you input the correct answer."

上面这个例题的说明是‘当 $yn 这个变数不是 "yes" 且 $yn 也不是 "YES" 时,才进行回圈内的程式。’ 而如果 $yn 是 "yes" 或 "YES" 时,就会离开回圈啰~那如果使用 until 呢?呵呵有趣啰~ 他的条件会变成这样:

[root@www scripts]# vi sh13-2.sh
#!/bin/bash
# Program:
#	Repeat question until user input correct answer.
# History:
# 2005/08/29	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

until [ "$yn" == "yes" -o "$yn" == "YES" ]
do
	read -p "Please input yes/YES to stop this program: " yn
done
echo "OK! you input the correct answer."

仔细比对一下这两个东西有啥不同喔! ^_^再来,如果我想要计算 1+2+3+....+100 这个数据呢? 利用回圈啊~他是这样的:

[root@www scripts]# vi sh14.sh
#!/bin/bash
# Program:
#	Use loop to calculate "1+2+3+...+100" result.
# History:
# 2005/08/29	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

s=0  # 这是加总的数值变数
i=0  # 这是累计的数值,亦即是 1, 2, 3....
while [ "$i" != "100" ]
do
	i=$(($i+1))   # 每次 i 都会增加 1 
	s=$(($s+$i))  # 每次都会加总一次!

done
echo "The result of '1+2+3+...+100' is ==> $s"

嘿嘿!当你执行了‘ sh sh14.sh ’之后,就可以得到 5050 这个数据才对啊!这样了呼~ 那么让你自行做一下,如果想要让使用者自行输入一个数字,让程式由 1+2+... 直到你输入的数字为止, 该如何撰写呢?应该很简单吧?答案可以参考一下习题练习里面的一题喔!


小标题的图示for...do...done (固定回圈)

相对于 while, until 的回圈方式是必须要‘符合某个条件’的状态, for 这种语法,则是‘ 已经知道要进行几次回圈’的状态!他的语法是:

for var in con1 con2 con3 ...
do

	程式段
done

以上面的例子来说,这个 $var 的变数内容在回圈工作时:

  1. 第一次回圈时, $var 的内容为 con1 ;
  2. 第二次回圈时, $var 的内容为 con2 ;
  3. 第三次回圈时, $var 的内容为 con3 ;
  4. ....

我们可以做个简单的练习。假设我有三种动物,分别是 dog, cat, elephant 三种, 我想每一行都输出这样:‘There are dogs...’之类的字样,则可以:

[root@www scripts]# vi sh15.sh

#!/bin/bash
# Program:
# 	Using for .... loop to print 3 animals
# History:
# 2005/08/29	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

for animal in dog cat elephant
do
	echo "There are ${animal}s.... "
done

等你执行之后就能够发现这个程式运作的情况啦!让我们想像另外一种状况,由于系统上面的各种帐号都是写在 /etc/passwd 内的第一个栏位,你能不能透过管线命令的 cut 捉出单纯的帐号名称后,以 idfinger 分别检查使用者的识别码与特殊参数呢?由于不同的 Linux 系统上面的帐号都不一样!此时实际去捉 /etc/passwd 并使用回圈处理,就是一个可行的方案了!程式可以如下:

[root@www scripts]# vi sh16.sh
#!/bin/bash
# Program
#       Use id, finger command to check system account's information.
# History
# 2009/02/18    VBird   first release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH
users=$(cut -d ':' -f1 /etc/passwd)  # 撷取帐号名称
for username in $users               # 开始回圈进行!
do
        id $username
        finger $username
done

执行上面的脚本后,你的系统帐号就会被捉出来检查啦!这个动作还可以用在每个帐号的删除、重整上面呢! 换个角度来看,如果我现在需要一连串的数字来进行回圈呢?举例来说,我想要利用 ping 这个可以判断网路状态的指令, 来进行网路状态的实际侦测时,我想要侦测的网域是本机所在的 192.168.1.1~192.168.1.100,由于有 100 台主机, 总不会要我在 for 后面输入 1 到 100 吧?此时你可以这样做喔!

[root@www scripts]# vi sh17.sh
#!/bin/bash
# Program
#       Use ping command to check the network's PC state.
# History
# 2009/02/18    VBird   first release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH
network="192.168.1"              # 先定义一个网域的前面部分!
for sitenu in $(seq 1 100)       # seq 为 sequence(连续) 的缩写之意
do
	# 底下的程式在取得 ping 的回传值是正确的还是失败的!

        ping -c 1 -w 1 ${network}.${sitenu} &> /dev/null && result=0 || result=1
	# 开始显示结果是正确的启动 (UP) 还是错误的没有连通 (DOWN)
        if [ "$result" == 0 ]; then
                echo "Server ${network}.${sitenu} is UP."
        else
                echo "Server ${network}.${sitenu} is DOWN."
        fi
done

上面这一串指令执行之后就可以显示出 192.168.1.1~192.168.1.100 共 100 部主机目前是否能与你的机器连通! 如果你的网域与鸟哥所在的位置不同,则直接修改上头那个 network 的变数内容即可!其实这个范例的重点在 $(seq ..) 那个位置!那个 seq 是连续 (sequence) 的缩写之意!代表后面接的两个数值是一直连续的! 如此一来,就能够轻松的将连续数字带入程式中啰!

最后,让我们来玩判断式加上回圈的功能!我想要让使用者输入某个目录档名, 然后我找出某目录内的档名的权限,该如何是好?呵呵!可以这样做啦~

[root@www scripts]# vi sh18.sh
#!/bin/bash
# Program:
#	User input dir name, I find the permission of files.
# History:
# 2005/08/29	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

# 1. 先看看这个目录是否存在啊?
read -p "Please input a directory: " dir
if [ "$dir" == "" -o ! -d "$dir" ]; then
	echo "The $dir is NOT exist in your system."
	exit 1
fi

# 2. 开始测试档案啰~
filelist=$(ls $dir)        # 列出所有在该目录下的档案名称

for filename in $filelist
do
	perm=""
	test -r "$dir/$filename" && perm="$perm readable"
	test -w "$dir/$filename" && perm="$perm writable"
	test -x "$dir/$filename" && perm="$perm executable"
	echo "The file $dir/$filename's permission is $perm "
done

呵呵!很有趣的例子吧~利用这种方式,你可以很轻易的来处理一些档案的特性呢。接下来,让我们来玩玩另一种 for 回圈的功能吧!主要用在数值方面的处理喔!


小标题的图示for...do...done 的数值处理

除了上述的方法之外,for 回圈还有另外一种写法!语法如下:

for (( 初始值; 限制值; 执行步阶 ))
do

	程式段
done

这种语法适合于数值方式的运算当中,在 for 后面的括号内的三串内容意义为:

  • 初始值:某个变数在回圈当中的起始值,直接以类似 i=1 设定好;
  • 限制值:当变数的值在这个限制值的范围内,就继续进行回圈。例如 i<=100;
  • 执行步阶:每作一次回圈时,变数的变化量。例如 i=i+1。

值得注意的是,在‘执行步阶’的设定上,如果每次增加 1 ,则可以使用类似‘i++’的方式,亦即是 i 每次回圈都会增加一的意思。好,我们以这种方式来进行 1 累加到使用者输入的回圈吧!

[root@www scripts]# vi sh19.sh
#!/bin/bash
# Program:
# 	Try do calculate 1+2+....+${your_input}
# History:
# 2005/08/29	VBird	First release
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
export PATH

read -p "Please input a number, I will count for 1+2+...+your_input: " nu

s=0
for (( i=1; i<=$nu; i=i+1 ))
do
	s=$(($s+$i))
done
echo "The result of '1+2+3+...+$nu' is ==> $s"

一样也是很简单吧!利用这个 for 则可以直接限制回圈要进行几次呢!


大标题的图示shell script 的追踪与 debug

scripts 在执行之前,最怕的就是出现语法错误的问题了!那么我们如何 debug 呢?有没有办法不需要透过直接执行该 scripts 就可以来判断是否有问题呢?呵呵!当然是有的!我们就直接以 bash 的相关参数来进行判断吧!

[root@www ~]# sh [-nvx] scripts.sh
选项与参数:
-n  :不要执行 script,仅查询语法的问题;
-v  :再执行 sccript 前,先将 scripts 的内容输出到荧幕上;
-x  :将使用到的 script 内容显示到荧幕上,这是很有用的参数!

范例一:测试 sh16.sh 有无语法的问题?
[root@www ~]# sh -n sh16.sh 
# 若语法没有问题,则不会显示任何资讯!

范例二:将 sh15.sh 的执行过程全部列出来~

[root@www ~]# sh -x sh15.sh 
+ PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:/root/bin
+ export PATH
+ for animal in dog cat elephant
+ echo 'There are dogs.... '
There are dogs....
+ for animal in dog cat elephant
+ echo 'There are cats.... '
There are cats....
+ for animal in dog cat elephant
+ echo 'There are elephants.... '
There are elephants....

请注意,上面范例二中执行的结果并不会有颜色的显示!鸟哥为了方便说明所以在 + 号之后的资料都加上颜色了! 在输出的讯息中,在加号后面的资料其实都是指令串,由于 sh -x 的方式来将指令执行过程也显示出来, 如此使用者可以判断程式码执行到哪一段时会出现相关的资讯!这个功能非常的棒!透过显示完整的指令串, 你就能够依据输出的错误资讯来订正你的脚本了!

熟悉 sh 的用法,将可以使你在管理 Linux 的过程中得心应手!至于在 Shell scripts 的学习方法上面,需要‘多看、多模仿、并加以修改成自己的样式!’ 是最快的学习手段了!网路上有相当多的朋友在开发一些相当有用的 scripts ,若是你可以将对方的 scripts 拿来,并且改成适合自己主机的样子!那么学习的效果会是最快的呢!

另外,我们 Linux 系统本来就有很多的服务启动脚本,如果你想要知道每个 script 所代表的功能是什么? 可以直接以 vim 进入该 script 去查阅一下,通常立刻就知道该 script 的目的了。 举例来说,我们之前一直提到的 /etc/init.d/syslog ,这个 script 是干嘛用的? 利用 vi 去查阅最前面的几行字,他出现如下资讯:

# description: Syslog is the facility by which many daemons use to log \
# messages to various system log files.  It is a good idea to always \
# run syslog.
### BEGIN INIT INFO
# Provides: $syslog
### END INIT INFO

简单的说,这个脚本在启动一个名为 syslog 的常驻程式 (daemon),这个常驻程式可以帮助很多系统服务记载她们的登录档 (log file), 我们的 Linux 建议你一直启动 syslog 是个好主意!嘿嘿!简单的看看您就知道啥是啥啦!

另外,本章所有的范例都可以在 http://linux.vbird.org/linux_basic/0340bashshell-scripts/scripts-v3.tar.bz2 里头找到喔!加油~


大标题的图示重点回顾
  • shell script 是利用 shell 的功能所写的一个‘程式 (program)’,这个程式是使用纯文字档,将一些 shell 的语法与指令(含外部指令)写在里面, 搭配正规表示法、管线命令与资料流重导向等功能,以达到我们所想要的处理目的
  • shell script 用在系统管理上面是很好的一项工具,但是用在处理大量数值运算上, 就不够好了,因为 Shell scripts 的速度较慢,且使用的 CPU 资源较多,造成主机资源的分配不良。
  • 在 Shell script 的档案中,指令的执行是从上而下、从左而右的分析与执行;
  • shell script 的执行,至少需要有 r 的权限,若需要直接指令下达,则需要拥有 r 与 x 的权限;
  • 良好的程式撰写习惯中,第一行要宣告 shell (#!/bin/bash) ,第二行以后则宣告程式用途、版本、作者等
  • 对谈式脚本可用 read 指令达成;
  • 要建立每次执行脚本都有不同结果的资料,可使用 date 指令利用日期达成;
  • script 的执行若以 source 来执行时,代表在父程序的 bash 内执行之意!
  • 若需要进行判断式,可使用 test 或中括号 ( [] ) 来处理;
  • 在 script 内,$0, $1, $2..., $@ 是有特殊意义的!
  • 条件判断式可使用 if...then 来判断,若是固定变数内容的情况下,可使用 case $var in ... esac 来处理
  • 回圈主要分为不定回圈 (while, until) 以及固定回圈 (for) ,配合 do, done 来达成所需任务!
  • 我们可使用 sh -x script.sh 来进行程式的 debug

大标题的图示本章习题
( 要看答案请将滑鼠移动到‘答:’底下的空白处,按下左键圈选空白处即可察看 )
底下皆为实作题,请自行撰写出程式喔!
  • 请建立一支 script ,当你执行该 script 的时候,该 script 可以显示: 1. 你目前的身份 (用 whoami ) 2. 你目前所在的目录 (用 pwd)
    #!/bin/bash
    echo -e "Your name is ==> $(whoami)"
    echo -e "The current directory is ==> $(pwd)"
  • 请自行建立一支程式,该程式可以用来计算‘你还有几天可以过生日’啊?
    #!/bin/bash
    read -p "Pleas input your birthday (MMDD, ex> 0709): " bir
    now=`date +%m%d`
    if [ "$bir" == "$now" ]; then
    echo "Happy Birthday to you!!!"
    elif [ "$bir" -gt "$now" ]; then
    year=`date +%Y`
    total_d=$(($((`date --date="$year$bir" +%s`-`date +%s`))/60/60/24))
    echo "Your birthday will be $total_d later"
    else
    year=$((`date +%Y`+1))
    total_d=$(($((`date --date="$year$bir" +%s`-`date +%s`))/60/60/24))
    echo "Your birthday will be $total_d later"
    fi
  • 让使用者输入一个数字,程式可以由 1+2+3... 一直累加到使用者输入的数字为止。
    #!/bin/bash
    read -p "Please input an integer number: " number
    i=0
    s=0
    while [ "$i" != "$number" ]
    do
    i=$(($i+1))
    s=$(($s+$i))
    done
    echo "the result of '1+2+3+...$number' is ==> $s"
  • 撰写一支程式,他的作用是: 1.) 先查看一下 /root/test/logical 这个名称是否存在; 2.) 若不存在,则建立一个档案,使用 touch 来建立,建立完成后离开; 3.) 如果存在的话,判断该名称是否为档案,若为档案则将之删除后建立一个目录,档名为 logical ,之后离开; 4.) 如果存在的话,而且该名称为目录,则移除此目录!
    #!/bin/bash
    if [ ! -e logical ]; then
    touch logical
    echo "Just make a file logical"
    exit 1
    elif [ -e logical ] && [ -f logical ]; then
    rm logical
    mkdir logical
    echo "remove file ==> logical"
    echo "and make directory logical"
    exit 1
    elif [ -e logical ] && [ -d logical ]; then
    rm -rf logical
    echo "remove directory ==> logical"
    exit 1
    else
    echo "Does here have anything?"
    fi
  • 我们知道 /etc/passwd 里面以 : 来分隔,第一栏为帐号名称。请写一只程式,可以将 /etc/passwd 的第一栏取出,而且每一栏都以一行字串‘The 1 account is "root" ’来显示,那个 1 表示行数。
    #!/bin/bash
    accounts=`cat /etc/passwd | cut -d':' -f1`
    for account in $accounts
    do
    declare -i i=$i+1
    echo "The $i account is \"$account\" "
    done

大标题的图示参考资料与延伸阅读

2002/06/27:第一次完成
2003/02/10:重新编排与加入 FAQ
2005/08/29:将旧的文章移动到 这里 了。
2005/08/29:呼呼~加入了一些比较有趣的练习题~比第一版要难的多~大家多多玩一玩喔~
2009/02/10:将旧的基于 FC4 版本的文章移动到此处
2009/02/17:加入 shift 的介绍
2009/02/18:加入了一些额外的练习,包括 for 的 ping 处理!



 
     
本网页主要以 firefox 配合解析度 1024x768 作为设计依据
http://linux.vbird.org is designed by VBird during 2001-2009. Aerosol Lab.