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

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

  1. Not exist <==结果显示不存在啊!

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

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

  • 这个文件是否存在,若不存在则给予一个“Filename does not exist”的讯息,并中断程序;
  • 若这个文件存在,则判断他是个文件或目录,结果输出“Filename is regular file”或 “Filename is directory”
  • 判断一下,执行者的身份对这个文件或目录所拥有的权限,并输出权限数据!
    你可以先自行创作看看,然后再跟下面的结果讨论讨论。注意利用 test 与 && 还有 || 等标志!
  1. [dmtsai@study bin]$ vim file_perm.sh
  2. #!/bin/bash
  3. # Program:
  4. # User input a filename, program will check the flowing:
  5. # 1.) exist? 2.) file/directory? 3.) file permissions
  6. # History:
  7. # 2015/07/16 VBird First release
  8. PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
  9. export PATH
  10. # 1\. 让使用者输入文件名,并且判断使用者是否真的有输入字串?
  11. echo -e "Please input a filename, I will check the filename's type and permission. \n\n"
  12. read -p "Input a filename : " filename
  13. test -z ${filename} && echo "You MUST input a filename." && exit 0
  14. # 2\. 判断文件是否存在?若不存在则显示讯息并结束脚本
  15. test ! -e ${filename} && echo "The filename '${filename}' DO NOT exist" && exit 0
  16. # 3\. 开始判断文件类型与属性
  17. test -f ${filename} && filetype="regulare file"
  18. test -d ${filename} && filetype="directory"
  19. test -w ${filename} && perm="${perm} writable"
  20. test -x ${filename} && perm="${perm} executable"
  21. echo "The filename: ${filename} is a ${filetype}"
  22. echo "And the permissions for you are : ${perm}"

如果你执行这个脚本后,他会依据你输入的文件名来进行检查喔!先看是否存在,再看为文件或目录类型,最后判断权限。 但是你必须要注意的是,由于 root 在很多权限的限制上面都是无效的,所以使用 root 执行这个脚本时, 常常会发现与 ls -l 观察到的结果并不相同!所以,建议使用一般使用者来执行这个脚本试看看。

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

  1. [dmtsai@study ~]$ [ -z "${HOME}" ] ; echo $?

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

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

  • 在中括号 [] 内的每个元件都需要有空白键来分隔;
  • 在中括号内的变量,最好都以双引号括号起来;
  • 在中括号内的常数,最好都以单或双引号括号起来。
    为什么要这么麻烦啊?直接举例来说,假如我设置了 name="VBird Tsai" ,然后这样判定:
  1. [dmtsai@study ~]$ name="VBird Tsai"
  2. [dmtsai@study ~]$ [ ${name} == "VBird" ]
  3. bash: [: too many arguments

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

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

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

  • 当执行一个程序的时候,这个程序会让使用者选择 Y 或 N ,
  • 如果使用者输入 Y 或 y 时,就显示“ OK, continue ”
  • 如果使用者输入 n 或 N 时,就显示“ Oh, interrupt !”
  • 如果不是 Y/y/N/n 之内的其他字符,就显示“ I don't know what your choice is ”
    利用中括号、 && 与 || 来继续吧!
  1. [dmtsai@study bin]$ vim ans_yn.sh
  2. #!/bin/bash
  3. # Program:
  4. # This program shows the user's choice
  5. # History:
  6. # 2015/07/16 VBird First release
  7. PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
  8. export PATH
  9. read -p "Please input (Y/N): " yn
  10. [ "${yn}" == "Y" -o "${yn}" == "y" ] && echo "OK, continue" && exit 0
  11. [ "${yn}" == "N" -o "${yn}" == "n" ] && echo "Oh, interrupt!" && exit 0
  12. echo "I don't know what your choice is" && exit 0

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

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

  1. [dmtsai@study ~]$ file /etc/init.d/network
  2. /etc/init.d/network: Bourne-Again shell script, ASCII text executable
  3. # 使用 file 来查询后,系统告知这个文件是个 bash 的可执行 script 喔!

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

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

  • $# :代表后接的参数“个数”,以上表为例这里显示为“ 4 ”;
  • $@ :代表“ "$1" "$2" "$3" "$4" ”之意,每个变量是独立的(用双引号括起来);
  • $ :代表“ "$1c$2c$3c$4" ”,其中 c 为分隔字符,默认为空白键, 所以本例中代表“ "$1 $2 $3 $4" ”之意。
    那个 $@ 与 $
    基本上还是有所不同啦!不过,一般使用情况下可以直接记忆 $@ 即可! 好了,来做个例子吧~假设我要执行一个可以携带参数的 script ,执行该脚本后屏幕会显示如下的数据:

  • 程序的文件名为何?

  • 共有几个参数?
  • 若参数的个数小于 2 则告知使用者参数数量太少
  • 全部的参数内容为何?
  • 第一个参数为何?
  • 第二个参数为何

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}"

执行结果如下:

  1. [dmtsai@study bin]$ sh how_paras.sh theone haha quot
  2. The script name is ==> how_paras.sh <==文件名
  3. Total parameter number is ==> 3 <==果然有三个参数
  4. Your whole parameter is ==> 'theone haha quot' <==参数的内容全部
  5. The 1st parameter ==> theone <==第一个参数
  6. The 2nd parameter ==> haha <==第二个参数
  • shift:造成参数变量号码偏移
    除此之外,脚本后面所接的变量是否能够进行偏移 (shift) 呢?什么是偏移啊?我们直接以下面的范例来说明好了, 用范例说明比较好解释!我们将 how_paras.sh 的内容稍作变化一下,用来显示每次偏移后参数的变化情况:
  1. [dmtsai@study bin]$ vim shift_paras.sh
  2. #!/bin/bash
  3. # Program:
  4. # Program shows the effect of shift function.
  5. # History:
  6. # 2009/02/17 VBird First release
  7. PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin:~/bin
  8. export PATH
  9. echo "Total parameter number is ==> $#"
  10. echo "Your whole parameter is ==> '$@'"
  11. shift # 进行第一次“一个变量的 shift ”
  12. echo "Total parameter number is ==> $#"
  13. echo "Your whole parameter is ==> '$@'"
  14. shift 3 # 进行第二次“三个变量的 shift ”
  15. echo "Your whole parameter is ==> '$@'"

这玩意的执行成果如下:

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

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