shell脚本心得

零零星星的接触到写一些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

来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/23718752/viewspace-1248184/,如需转载,请注明出处,否则将追究法律责任。

转载于:http://blog.itpub.net/23718752/viewspace-1248184/

你可能感兴趣的:(shell脚本心得)