Linux shell脚本基础学习详细介绍(完整版)二

详细介绍Linux shell脚本基础学习(五) Linux shell脚本基础前面咱们在介绍Linux shell脚本的控制流程时,还有一部份内容没讲就是有关here document的内容这里继续。 
Linux shell脚本基础已经被分红好几个部分了,这里对控制流程的内容也就立刻讲完了,这是最后一部分关于here document,这里举例稍微有点复杂,咱们慢慢来分析这个复杂Linux shell脚本。html

 

6. Here documents程序员

当要将几行文字传递给一个命令时,here documents(译者注:目前尚未见到过对该词适合的翻译)一种不错的方法。对每一个脚本写一段帮助性的文字是颇有用的,此时若是咱们四有那个 here documents就没必要用echo函数一行行输出。 一个 "Here document" 以 << 开头,后面接上一个字符串,这个字符串还必须出如今here document的末尾。下面是一个例子,在该例子中,咱们对多个文件进行重命名,而且使用here documents打印帮助:算法

#!/bin/shshell

# we have less than 3 arguments. Print the help text:express

if [ $# -lt 3 ] ; then编程

cat <vim

ren -- renames a number of files using sed regular expressions框架

USAGE: ren 'regexp' 'replacement' files...less

EXAMPLE: rename all *.HTM files in *.html:ide

ren 'HTM$' 'html' *.HTM

HELP

exit 0

fi

OLD="$1"

NEW="$2"

# The shift command removes one argument from the list of

# command line arguments.

shift

shift

# $* contains now all the files:

for file in $*; do

if [ -f "$file" ] ; then

newfile=`echo "$file" | sed "s/${OLD}/${NEW}/g"`

if [ -f "$newfile" ]; then

echo "ERROR: $newfile exists already"

else

echo "renaming $file to $newfile ..."

mv "$file" "$newfile"

fi

fi

done

这是一个复杂一些的例子。让咱们详细讨论一下。第一个if表达式判断输入命令行参数是否小于3个 (特殊变量$# 表示包含参数的个数) 。若是输入参数小于3个,则将帮助文字传递给cat命令,而后由cat命令将其打印在屏幕上。打印帮助文字后程序退出。若是输入参数等于或大于3个,咱们 就将第一个参数赋值给变量OLD,第二个参数赋值给变量NEW。下一步,咱们使用shift命令将第一个和第二个参数从参数列表中删除,这样原来的第三个 参数就成为参数列表$*的第一个参数。而后咱们开始循环,命令行参数列表被一个接一个地被赋值给变量$file。接着咱们判断该文件是否存在,若是存在则 经过sed命令搜索和替换来产生新的文件名。而后将反短斜线内命令结果赋值给newfile。这样咱们就达到了咱们的目的:获得了旧文件名和新文件名。然 后使用mv命令进行重命名。这样就明了这个复杂的Linux shell脚本了吧。

详细介绍Linux shell脚本基础学习(六) Linux shell脚本基础学习咱们这里就差很少讲完了,最后一部份内容是关于函数的,这就差很少把基础部分介绍完了,后面还会有实例。 

4)函数

若是您写了一些稍微复杂一些的程序,您就会发如今程序中可能在几个地方使用了相同的代码,而且您也会发现,若是咱们使用了函数,会方便不少。一个函数是这个样子的:

functionname()

{

# inside the body $1 is the first argument given to the function

# $2 the second ...

body

}

您须要在每一个程序的开始对函数进行声明。

下面是一个叫作xtitlebar的脚本,使用这个脚本您能够改变终端窗口的名称。

这里使用了一个叫作help的函数。正如您能够看到的那样,这个定义的函数被使用了两次。

#!/bin/sh

# vim: set sw=4 ts=4 et:

help()

{

cat <

xtitlebar -- change the name of an xterm, gnome-terminal or kde konsole

USAGE: xtitlebar [-h] "string_for_titelbar"

OPTIONS: -h help text

EXAMPLE: xtitlebar "cvs"

HELP

exit 0

}

# in case of error or if -h is given we call the function help:

[ -z "$1" ] && help

[ "$1" = "-h" ] && help

# send the escape sequence to change the xterm titelbar:

echo -e "33]0;$107"

#

在脚本中提供帮助是一种很好的编程习惯,这样方便其余用户(和您)使用和理解脚本。

命令行参数

咱们已经见过$* 和 $1, $2 ... $9 等特殊变量,这些特殊变量包含了用户从命令行输入的参数。迄今为止,咱们仅仅了解了一些简单的命令行语法(好比一些强制性的参数和查看帮助的-h选项)。 可是在编写更复杂的程序时,您可能会发现您须要更多的自定义的选项。一般的惯例是在全部可选的参数以前加一个减号,后面再加上参数值 (好比文件名)

有好多方法能够实现对输入参数的分析,可是下面的使用case表达式的例子无遗是一个不错的方法。

#!/bin/sh

help()

{

cat <

This is a generic command line parser demo.

USAGE EXAMPLE: cmdparser -l hello -f -- -somefile1 somefile2

HELP

exit 0

}

while [ -n "$1" ]; do

case $1 in

-h) help;shift 1;; # function help is called

-f) opt_f=1;shift 1;; # variable opt_f is set

-l) opt_l=$2;shift 2;; # -l takes an argument -> shift by 2

--) shift;break;; # end of options

-*) echo "error: no such option $1. -h for help";exit 1;;

*) break;;

esac

done

echo "opt_f is $opt_f"

echo "opt_l is $opt_l"

echo "first arg is $1"

echo "2nd arg is $2"

您能够这样运行该脚本:

cmdparser -l hello -f -- -somefile1 somefile2

返回的结果是:

opt_f is 1

opt_l is hello

first arg is -somefile1

2nd arg is somefile2

这个脚本是如何工做的呢?脚本首先在全部输入命令行参数中进行循环,将输入参数与case表达式进行比较,若是匹配则设置一个变量而且移除该参数。根据unix系统的惯例,首先输入的应该是包含减号的参数。

详细介绍Linux shell脚本基础学习(七) Linux shell脚本基础的学习理论知识已经讲完了,后面是两个具体的实例,这里先说第一个关于二进制什么时候禁止之间的转换。 
Linux shell脚本基础学习这部分若是只看前面间的理论部分虽然有一些例子,可是还不够系统,这里将以具体实例给你们展示Linux shell脚本编程,以帮助你们完善Linux shell基础的学习和提升。
第2部分 实例
如今咱们来讨论编写一个脚本的通常步骤。任何优秀的脚本都应该具备帮助和输入参数。而且写一个伪脚本(framework.sh),该脚本包含了大多数脚本都须要的框架结构,是一个很是不错的主意。这时候,在写一个新的脚本时咱们只须要执行一下copy命令:
cp framework.sh myscript
而后再插入本身的函数。
让咱们再看个例子:
二进制到十进制的转换
脚本 b2d 将二进制数 (好比 1101) 转换为相应的十进制数。这也是一个用expr命令进行数学运算的例子:

复制代码 代码以下:
#!/bin/sh # vim: set sw=4 ts=4 et:
help() { cat < b2h -- convert binary to decimal USAGE: b2h [-h] binarynum OPTIONS: -h help text EXAMPLE: b2h 111010 will return 58 HELP exit 0 } error() { # print an error and exit echo "$1" exit 1 } lastchar() { # return the last character of a string in $rval if [ -z "$1" ]; then # empty string
rval="" return fi # wc puts some space behind the output this is why we need sed: numofchar=`echo -n "$1" | wc -c | sed 's/ //g' ` # now cut out the last char rval=`echo -n "$1" | cut -b $numofchar` }
chop() { # remove the last character in string and return it in $rval if [ -z "$1" ]; then # empty string rval="" return fi
# wc puts some space behind the output this is why we need sed:
numofchar=`echo -n "$1" | wc -c | sed 's/ //g' ` if [ "$numofchar" = "1" ]; then # only one char in string rval="" return fi
numofcharminus1=`expr $numofchar "-" 1` # now cut all but the last char:
rval=`echo -n "$1" | cut -b 0-${numofcharminus1}` } while [ -n "$1" ]; do case $1 in -h) help;shift 1;; # function help is called --) shift;break;; # end of options -*) error "error: no such option $1. -h for help";; *) break;; esac done # The main program sum=0
weight=1 # one arg must be given: [ -z "$1" ] && help
binnum="$1" binnumorig="$1" while [ -n "$binnum" ]; do lastchar "$binnum" if [ "$rval" = "1" ]; then sum=`expr "$weight" "+" "$sum"`
fi # remove the last position in $binnum chop "$binnum"
binnum="$rval" weight=`expr "$weight" "*" 2` done echo "binary $binnumorig is decimal $sum"

该脚本使用的算法是利用十进制和二进制数权值 (1,2,4,8,16,..),好比二进制"10"能够这样转换成十进制:
0 * 1 + 1 * 2 = 2
为了获得单个的二进制数咱们是用了lastchar 函数。该函数使用wc –c计算字符个数,而后使用cut命令取出末尾一个字符。Chop函数的功能则是移除最后一个字符。
这个Linux shell脚本实例帮咱们完成了转换,下一次咱们将举例一个文件循环程序。 
详细介绍Linux shell脚本基础学习(八)
Linux shell脚本基础学习实例前面说明了十进制和二进制的转换,这里举最后一个例子,关于文件的循环,同时也说明一下如何调试,来结束咱们的课程。 

 

Linux shell脚本前面的实例是说明十进制和二进制的转换,还以一个有关文件循环的实例来结束这部份内容的学习。相信Linux shell脚本的基础学习的学习者应该可以掌握一些简单的Linux shell脚本的编写。 文件循环程序

或许您是想将全部发出的邮件保存到一个文件中的人们中的一员,可是在过了几个月之后,这个文件可能会变得很大以致于使对该文件的访问速度变慢。下面的 脚本rotatefile可 以解决这个问题。这个脚本能够重命名邮件保存文件(假设为outmail)为outmail.1,而对于 outmail.1就变成了outmail.2 等等等等...

 

复制代码 代码以下:
#!/bin/sh # vim: set sw=4 ts=4 et: ver="0.1" help() { cat < rotatefile -- rotate the file name USAGE: rotatefile [-h] filename OPTIONS: -h help text EXAMPLE: rotatefile out This will e.g rename out.2 to out.3, out.1 to out.2, out to out.1 and create an empty out-file The max number is 10 version $ver HELP exit 0 } error() { echo "$1" exit 1 } while [ -n "$1" ]; do case $1 in -h) help;shift 1;; --) break;; -*) echo "error: no such option $1. -h for help";exit 1;; *) break;; esac done # input check: if [ -z "$1" ] ; then error "ERROR: you must specify a file, use -h for help" fi filen="$1" # rename any .1 , .2 etc file: for n in 9 8 7 6 5 4 3 2 1; do if [ -f "$filen.$n" ]; then p=`expr $n + 1` echo "mv $filen.$n $filen.$p" mv $filen.$n $filen.$p fi done # rename the original file: if [ -f "$filen" ]; then echo "mv $filen $filen.1" mv $filen $filen.1 fi echo touch $filen touch $filen

 

这个脚本是如何工做的呢?在检测用户提供了一个文件名之后,咱们进行一个9到1的循环。文件9被命名为10,文件8重命名为9等等。循环完成以后,咱们将原始文件命名为文件1 同时创建一个与原始文件同名的空文件。

调试

最简单的调试命令固然是使用echo命令。您可使用echo在任何怀疑出错的地方打印任何变量值。这也是绝大多数的shell程序员要花费80%的时间来调试程序的缘由。Shell程序的 好处在于不须要从新编译,插入一个echo命令也不须要多少时间。

shell也有一个真实的调试模式。若是在脚本"strangescript" 中有错误,您能够这样来进行调试:

sh -x strangescript

这将执行该脚本并显示全部变量的值。

shell还有一个不须要执行脚本只是检查语法的模式。能够这样使用:

sh -n your_script

这将返回全部语法错误

相关文章
相关标签/搜索