零零星星的接触到写一些shell也有一些日子了,发现自己已经犯了不少的错误,自我总结下。
选择合适的shell
shell本身有很多种,大体有如下的几种。
/bin/sh (已经被 /bin/bash 所取代)
/bin/bash (就是 Linux 预设的 shell)
/bin/ksh (Kornshell 由 AT&T Bell lab. 发展出來的,相融于 bash)
/bin/tcsh (整合 C Shell ,提供更多的功能)
/bin/csh (已经被 /bin/tcsh 所取代)
/bin/zsh (基于 ksh 发展出來的,功能更強大的 shell)
可以根据工作的需要和自己的实际情况来选择,目前我使用比较多的就是bash,ksh
不同路径执行,不会影响结果
这个问题比较纠结,在自己写的一些脚本中,没有注意到一些路径的设置,可能在当前目录下执行脚本和在其他路径下执行就有很大的差别,甚至是严重的错误。
比如我现在有一个脚本test.sh在目录 /u01/ora11g下面。
那么我在/u01/ora11g下面执行自然没有问题,但是比如我现在在/u02/db2的目录下面,我想运行这个test.sh 可能就需要输入 ksh /u01/ora11g/test.sh xxxxx
如果路径的一些通用性没有考虑到的话,这个脚本很可能出错,或者出现不期望的结果。
临时文件的处理和命名
对于临时文件的处理,个人建议统一命名,比如以以下特定的操作,功能命名。
xxxx_rename_file.tmp等等。
命令的简化和功能的简化
对于这个部分,需要大家自己把握一个度,可能有些人喜欢用一个很“精简”的命令来完成一个很复杂的工作。有些人喜欢通过一些很简单的操作来组合起来,完成一个复杂的功能。
但是需要大家自己把握这个度,如果脚本的维护和后期的改进不是你一个人来完成的话,最好还是以一些规范为主,适当加入一些相关的注释和说明。有些精简的命令可以加一些特定的描述,这样在后期需要改进,就很容易把握。
适用的平台
如果大家在unix,linux下写过一些脚本,可能会发现有一些命令的选项在Linux可用,但是到了Unix下却并不买账。
比如awk,grep在SunOS,AIX,linux下对应的路径有很大的差别,如果想让命令更通用,可以考虑下面的形式。
if [ $MachineType = SunOS ]
then
export AWK=/usr/xpg4/bin/awk
export GREP=/usr/xpg4/bin/grep
export SED=/usr/xpg4/bin/sed
export TR=/usr/xpg4/bin/tr
export RM=/usr/xpg4/bin/rm
export CAT=/usr/bin/cat
export MV=/usr/xpg4/bin/mv
export CUT=/usr/bin/cut
export WHOAMI=/usr/ucb/whoami
export SORT=/usr/xpg4/bin/sort
export CP=/usr/xpg4/bin/cp
elif [ $MachineType = AIX ]
then
export AWK=/usr/bin/awk
export GREP=/usr/local/bin/grep
export SED=/usr/bin/sed
export TR=/usr/bin/tr
export RM=/usr/bin/rm
export CAT=/usr/bin/cat
export MV=/usr/bin/mv
export CUT=/usr/bin/cut
export WHOAMI=/usr/bin/whoami
export SORT=/usr/bin/sort
export CP=/usr/bin/cp
elif [ $MachineType = Linux ]
then
export AWK=/bin/awk
export GREP=/bin/grep
export SED=/bin/sed
export TR=/usr/bin/tr
export RM=/bin/rm
export CAT=/bin/cat
export MV=/bin/mv
export CUT=/bin/cut
。。。。。
日志
对于脚本中的数据,文件处理,最好还是有一些详尽的日志,没有日志,谁也不知道到底发生了什么。
而且对于问题的排查时极为重要的。
函数库
如果你已经沉淀了不少的功能集,可以考虑把他们整合到函数库中,在以后的处理中直接调用即可。
动态脚本
完成一些复杂的功能时,可以考虑使用动态脚本来实现。
可以考虑通过shell脚本来生成一些特定功能的shell脚本。
比如:使用动态变量进行动态数据比较
http://blog.itpub.net/23718752/viewspace-1210639/
完整的数据校验和容错处理
脚本的编写过程中,可能大家经常忽略的就是一些数据的校验功能,可能有很多细节都没有做校验,在复杂的场景中就很容易出错,如果要写一个比较完善的脚本,那么数据的校验和错误的处理都是需要格外关注的,毕竟软件的很多细节都是成败的关键。
强大的工具集 sed+awk
sed+awk在shell脚本的编写中有很重要的作用,使用的过程中正则表达式的一些知识也需要补补。
有很多的功能可能通过一些文件处理能够实现,但是有时候就很容易使用sed/awk来完成。
抛砖引用一下,比如我想对当前目录下的文件,输出文件名都添加一个后缀.abc
可以这样来写
ls -l|awk ‘{print $9 “.abc”}’
交互性
可能在写脚本的时候,大家对于一些参数需要输入,就可以提供一些可读性的提示信息。这样在操作的时候更容易理解。
可以使用read来引入一些输入参数的值,加入一些提示信息。
cat test.sh
echo 'please input your message:'; read name
echo 'your message is :'$name
[ora11g@rac1 ~]$ ksh testa.sh
please input your message:
this is a test
your message is :this is a test
充分利用其他的工具的功能集
使用shell做数据的运算,和其他编程语言相比,感觉还是比较的吃力,比如我想做一个舍入的运算,在oracle中就是ceil()函数实现的功能。
使用一个简单的sql就马上得到期望的结果,。
SQL> select ceil(100/3) from dual;
CEIL(100/3)
-----------
34
但是使用shell需要类似下面的一些转换和处理,当然了在文件的处理方面,shell的功能很强大。
pages_float=`echo "scale=2 ; 100 / 3 "|bc`
pages_num=`echo '' | awk -v a=$pages_float '{print int(a+0.999)}'`
echo $pages_num
34