bash脚本编程class3

bash脚本编程class3—函数定义与数组使用相关

    一.函数的使用以及数组的定义

  我们通过学习知道,当编辑一个shell脚本的时候,可以在这个shell脚本中调用其他的shell脚本来实现特定的功能。在实际工作环境中,用很多shell实现的功能是我们常常使用到的,如果一次一次的调用这些shell脚本会显得非常麻烦。这时,函数的作用就体现出来了。其实函数就是一个被定义并命名的常用shell脚本,所有函数都集中放在一个配置文件中,当用户使用的时候,只需要调用这个文件内容,就可以根据其中每个函数名称来使用。比如如果我们在/etc/functions这个文件写入了很多函数,如果想要在一个脚本中使用这些函数,只需要在脚本中调用/etc/functions这个文件,就可以使用这些函数了。

  

  1.函数的定义方式

  函数的定义方式有三种:

  function  函数名() { ;函数内容 ;}

  函数名 () { ;函数内容 ;}

  function 函数名 { ;函数内容 ;}

filejudge () {
[ ! -e $1 ]&&echo "meiwenjian"&& exit
if [[ -f $1 ]]
    then
    echo "This is a ordinary file"
elif [[ -d $1 ]]
    then
    echo "This is a directory"
elif [[ -h $1 ]]
    then
    echo "This is a link file"
else
    echo "This is an other file type"
fi
}

 上面使用的就是第二种函数命名方式,这个函数体实现的功能为辨别文件的类型。

  2.载入函数以及检验载入函数

  当我们在一个shell文件中编译好了一系列的函数之后,如何使用这些函数呢?这里需要使用source或者.命令,这样可以使函数文件的内容立即生效。使用set命令可以查看当前系统中定义的所有函数。如果想取消某些函数的使用,使用unset 函数名,就可以取消这个函数的使用。载入函数以后,直接使用函数名[后加参数],就可以实现函数功能了。

  3.函数的返回值以及局部变量

  在执行一个命令之后,echo $?可以显示命令的成功与否。而函数大多是由多条命令组成的,正常情况下,当用户鉴定函数体是否执行成功时,是根据函数中最后一条命令的成功与否来判断的。如果想要定义返回值,使用return后接数字来实现。return 0表示成功,return1 1-255表示有错误返回值。同样使用echo命令来显示。return的好处是可以根据返回状态值进行条件选择执行不同分支的命令,从而使函数应用性更强。

  在函数中可以定义变量,可以定义本地变量与环境变量。但是如果用户只想要在函数中使用这个变量,执行函数完成后不想让这个变量在当前shell中继续使用,防止与同名变量冲突。这该怎么办呢?Linux的有一种变量可以达到这个卸磨杀驴的效果,就是局部变量,这个变量只能在函数中定义,只能在函数体中使用,执行完函数体就没效果了。使用local name=变量名,来定义。比如:

#!/bin/bash
name=dafei
xxx () {
   local name=xiaowangba
}
echo $name

 上述我们写一个函数的脚本,先定义变量name=dafei,然后在函数体内定义局部变量name=xiaowangba,在函数外执行name变量显示。如果执行这个脚本,你猜name会是什么?

[root@localhost bin]# bash funstest.sh
dafei

 当然是dafei。

  4.函数递归以及其使用实例

  函数递归说白了就是函数自己调用自己,前提是此函数可以使用的情况下。应用这种递归方式,可以实现一些规律性的数据计算。比如 Fn = Fn-1 + Fn-2。如果想计算Fn的值,需要如下函数递归:

goldrabbit () {
  if [ $1 -eq 1 ];then
     echo 0
  elif [ $1 -eq 2 -o $1 -eq 3 ];then
     echo 1
  else
     echo $[$(goldrabbit $[$1-1])+$(goldrabbit $[$1-2])]
  fi
}

 goldrabbit为函数名,根据数列规律我们得知:F3=F2+F1=1,F2=F3=1,F1=0。F4=F2+F3=2…这样就可以得知Fn的值了。

    二.数组定义以及字符串处理

  

  1.数组的定义

  之前所学的变量每一次赋值都是单个数字或者字符串,如果想要一次赋值多个变量,就需要数组的概念了。使用name=(若干变量…)。定义name为一个数组。

  使用数组之前一般需要declare -a name声明数组。变量分割符为空格,默认从name[0]到name[n-1]。比如:

[root@localhost ~]# baobao=(xx yy zz aa bb)
[root@localhost ~]# echo ${baobao[1]}
yy

 如上这里为变量baobao定义一个数组,显示baobao[1],结果为数组中的第二个变量yy。

 有些时候数组的排序不适用0到n-1,使用a,b,c..也是可以的,但是前面一定要声明declare -A。比如:

[root@localhost ~]# declare -A baobao
[root@localhost ~]# baobao=([a]=aa [b]=bb [c]=cc [d]=dd)
[root@localhost ~]# echo ${baobao[b]}
bb

 这里可以看到数组变量的排序不是0到n-1了,而是a,b,c,d。这就是关联数组declare -A的作用。

  2.数组变量的修改

  实现数组内容的增删查改:只能追增,根据编号顺序追增,如果编号顺序重复则覆盖。删,使用unset name[number]删除编号的内容。查,echo ${name[*]}显示所有变量,echo ${#name[*]}显示数组中变量个数。上例子:

[root@localhost ~]# declare -a baobao
[root@localhost ~]# baobao=(aa bb cc dd ee)
[root@localhost ~]# echo ${#baobao[*]}
5
[root@localhost ~]# echo ${baobao[*]}
aa bb cc dd ee
[root@localhost ~]# baobao[6]=xx
[root@localhost ~]# echo ${baobao[*]}
aa bb cc dd ee xx
[root@localhost ~]# unset baobao[0]
[root@localhost ~]# echo ${baobao[*]}
bb cc dd ee xx

 增删查改baobao。

  也可以使用${baobao[*]:1:3}表示去掉前一个(按编号),然后接着取3个(按编号)baobao数组的变量:

echo ${baobao[*]:1:3}
bb cc dd

  由于baobao[0]被删除了,所以越过baobao[0]取接下来的baobao[1],baobao[2],baobao[3],于是得到bb cc dd

例题:使用冒泡排序法找出10个随机数的最大值

[root@localhost bin]# cat funstest.sh
#!/bin/bash
for i in `seq 0 9`;do
    ran[$i]=$RANDOM
    if [ $i -eq 0 ];then
       max=${ran[$i]}
    elif [ ${ran[$i]} -gt ${ran[$[$i-1]]} ];then
       max=${ran[$i]}
    fi
done
echo ${ran[@]}
echo "max: $max"
    

[root@localhost bin]# bash funstest.sh
3232 5655 22990 30240 28066 15611 11885 7793 15777 13386
max: 15777

  使用数组,非常简单。

  3.字符串处理

  这里指的字符串处理是bash自带功能的字符串处理,而不是通过cut命令之类的,同时字符串一般是由一个变量代表的。比如:

[root@localhost ~]# baobao=aaadfjaojsfpajfiqfkangiuowang
[root@localhost ~]# echo ${baobao:3}
dfjaojsfpajfiqfkangiuowang
[root@localhost ~]# echo ${baobao:3:2}
df

  上述命令,第一个表示取baobao去掉前三个字符的字符串内容,后一个表示去掉前三个接着只取后续的两个字符。

[root@localhost bin]# echo $baobao 
aaadfjaojsfpajfiqfkangiuowang
[root@localhost bin]# echo ${baobao#*js}
fpajfiqfkangiuowang
[root@localhost bin]# echo ${baobao%js*}
aaadfjao

  上述命令#,%分别代表选取字符串的特定内容,#表示从左向右找到js以及js右侧的内容,%表示从右向左找到js以及js左侧的内容,这里#,%后面的word表示第一次找到的word,如果换成##,%%就会变成最后一次找到的word

[root@localhost bin]# baobao=dadsbabagwegwjkhwoibabakjrgjwp
[root@localhost bin]# echo ${baobao##*baba}
kjrgjwp
[root@localhost bin]# echo ${baobao%%*baba}
dadsbabagwegwjkhwoibabakjrgjwp

 如上所述。

  最后介绍转换字符串大小写的方法:

[root@localhost bin]# baobao=dadsbabagwegwjkhwoibabakjrgjwp
[root@localhost bin]# echo ${baobao^^} 
DADSBABAGWEGWJKHWOIBABAKJRGJWP

[root@localhost bin]# xx=`echo ${baobao^^}`
[root@localhost bin]# echo $xx
DADSBABAGWEGWJKHWOIBABAKJRGJWP
[root@localhost bin]# echo ${xx,,}
dadsbabagwegwjkhwoibabakjrgjwp

  懂没?不懂再学。

原创文章,作者:mcgeewang,如若转载,请注明出处:http://www.178linux.com/39242

(0)
mcgeewangmcgeewang
上一篇 2016-08-24
下一篇 2016-08-24

相关推荐

  • 硬盘分区及挂载

    标签:文件系统、分区、挂载 一、Linux的基本原则    1、一切皆文件(包括硬件);这个原则会会在很多方面得到体现; 磁盘在Linux中也表现为文件,即/dev目录下:IDE,ATA:/dev/hd[a-z]    SATA,SCSI,USB,SAS:sd[a-z]。    &nbsp…

    Linux干货 2015-05-18
  • N25-第十周

    1、请详细描述CentOS系统的启动流程(详细到每个过程系统做了哪些事情) 一、内核空间详细过程如下: POST –> BootSequence(BIOS) –> Bootloader(MBR) –> kernel(ramdisk) –> rootfs(只读) –> /sbin/init(systemd) 1.POST:(pow…

    Linux干货 2017-05-30
  • 八大排序算法

    概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部的排序记录,在排序过程中需要访问外存。 我们这里说说八大排序就是内部排序。     当n较大,则应采用时间复杂度为O(nlog2n)的排序方法:快速排序、堆排序或归并排序序。    快速…

    Linux干货 2015-04-07
  • N24-vt.lee-学习宣言

    学习宣言:  好好学习 天天向上!!!  2016-10-26

    Linux干货 2016-10-26
  • 硬链接与软链接

    引言: 链接,一种在共享文件和访问它的用户的若干目录项之间建立联系的一种方法。 Linux中包括两种链接:硬链接(Hard Link)和软链接(Soft Link),软链接又称为符号链接(Symbolic link)。 Inode译成中文就是索引节点,它用来存放档案及目录的基本信息,包含时间、档名、使用者及群组等。 Block:存放档案内容数据的地方。 &n…

    Linux干货 2016-10-20

评论列表(1条)

  • 马哥教育
    马哥教育 2016-08-25 13:29

    文章整体思路清晰,shell脚本函数和数组定义理解透彻,看完博客给人一中概览性的认识,整篇博客让人能看到很多专业性所在,还能看出很用心的地方,革命尚未成功,同志仍需努力啊,加油!