shell脚本进阶

Posted 绮梦小煊

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了shell脚本进阶相关的知识,希望对你有一定的参考价值。

select 循环与菜单

select variable in list

   do

     循环体命令

  done

select 循环主要用于创建菜单,按数字顺序排列的菜单项将显示在标准错误上,并显示PS3 提示符,等待用户输入

 

用户输入菜单列表中的某个数字,执行相应的命令

用户输入被保存在内置变量REPLY 中。

 

select 是个无限循环,因此要记住用break 命令退出循环,或用exit 命令终止脚本。也可以按ctrl+c 退出循环。

 

select 经常和case 联合使用

与for 循环类似,可以省略in list ,此时使用位置参量

#!/bin/bash
select choice in 1yuan 2yuan 5yuan Quit ;do 
    case $choice in
        1yuan)
            echo "You can buy a glass of water "
            ;;  
        2yuan)
            echo "You can buy  an ice cream "
            ;;  
        5yuan)
            echo "You can buy  a chicken leg "
            echo "Choice $REPLY" 
            ;;  
        Quit)
            echo "Bye"
            break
            ;;  
        *)  
            echo "Enter error!"
            exit 2
      
    esac
done

技术分享

技术分享

技术分享

 

 

函数介绍

函数function是由若干条shell命令组成的语句块,实现代码重用和模块化编程。

它与shell程序形式上是相似的,不同的是它不是一个单独的进程,不能独立运行,而是shell程序的一部分。

函数和shell程序比较相似,区别在于:

  Shell程序在子Shell中运行

  而Shell函数在当前Shell中运行。因此在当前Shell中,函数可以对shell中变量进行修改

 

定义函数格式:

 

函数由两部分组成:函数名和函数体。

语法一:

  function f_name{

    …函数体…

  }

语法二:

  function f_name(){

    …函数体…

  }

语法三:

  f_name(){

    …函数体…

  }

 

 

函数使用:

函数的定义和使用:

    可在交互式环境下定义函数

    可将函数放在脚本文件中作为它的一部分

    可放在只包含函数的单独文件中

 

调用:函数只有被调用才会执行;

    调用:给定函数名

    函数名出现的地方,会被自动替换为函数代码

 

函数的生命周期:被调用时创建,返回时终止

 

函数返回值:

函数有两种返回值:

  函数的执行结果返回值:

    (1) 使用echo或printf命令进行输出

    (2) 函数体中调用命令的输出结果

  函数的退出状态码:

    (1) 默认取决于函数中执行的最后一条命令的退出状态码

    (2) 自定义退出状态码,其格式为:

        return 从函数中返回,用最后状态命令决定返回值

        return 0 无错误返回。

        return 1-255 有错误返回

 

交互式环境下定义和使用函数:

示例:

$dir() {
ls-l
> }

定义该函数后,若在$后面键入dir,其显示结果同ls-l的作用相同。

    $dir

该dir函数将一直保留到用户从系统退出,或执行了如下所示的unset命令:

    $ unsetdir

 

在脚本中定义及使用函数:

函数在使用前必须定义,因此应将函数定义放在脚本开始部分,直至shell首次发现它后才能使用

调用函数仅使用其函数名即可。

#!/bin/bash
#fun1
 
hello () {
    echo "Hello there today‘s date is `date +%F`"
}
 
echo "Now going to the funtion hello"
hello
echo "back from the function"

技术分享

 

使用函数文件

1、可以将经常使用的函数存入函数文件,然后将函数文件载入shell。

2、文件名可任意选取,但最好与相关任务有某种联系。例如:functions.main

3、一旦函数文件载入shell,就可以在命令行或脚本中调用函数。可以使用set命令查看所有定义的函   数,其输出列表包括已经载入shell的所有函数。

4、若要改动函数,首先用unset命令从shell中删除函数。改动完毕后,再重新载入此文件。

载入函数

函数文件已创建好后,要将它载入shell

定位函数文件并载入shell的格式:

    . filename 或source filename

   注意:此即<点> <空格> <文件名>

这里的文件名要带正确路径

检查载入函数

使用set命令检查函数是否已载入。set命令将在shell中显示所有的载入函数。

执行函数

要执行函数,简单地键入函数名即可

 

删除shell函数

现在对函数做一些改动。首先删除函数,使其对shell不可用。使用unset命令完成此功能.

命令格式为:unset function_name


函数参数:

函数可以接受参数:

    传递参数给函数:调用函数时,在函数名后面以空白分隔给定参数列表即可;

        例如“testfuncarg1 arg2 …”

    在函数体中当中,可使用$1, $2, …调用这些参数;还可以使用[email protected], $*, $#等特殊变量

函数变量

变量作用域:

  环境变量:当前shell和子shell有效

  本地变量:只在当前shell进程有效,为执行脚本会启动专用子shell进程;因此,本地变量的作用范         围是当前shell脚本程序文件,包括脚本中的函数。

  局部变量:函数的生命周期;函数结束时变量被自动销毁

注意:如果函数中有局部变量,如果其名称同本地变量,使用局部变量。

在函数中定义局部变量的方法:local NAME=VALUE

函数递归:函数直接或间接调用自身(注意递归层数)

递归实例:

阶乘是基斯顿·卡曼于1808 年发明的运算符号,是数学术语,一个正整数的阶乘(factorial)是所有小于及等于该数的正整数的积,并且有0的阶乘为1。自然数n的阶乘写作n!。

n!=1×2×3×…×n。

阶乘亦可以递归方式定义:0!=1,n!=(n-1)!×n。

n!=n(n-1)(n-2)…1

n(n-1)! = n(n-1)(n-2)!

#!/bin/bash
jiecheng () {
    if [ $1 -eq 1 -o $1 -eq 0 ] ;then
        echo 1
    else
        echo "$[$1*$(jiecheng $[$1-1])]"
    fi  
}
 
read -p "计算阶乘,请输入一个正数字:" n
if [[ $n =~ ^[[:digit:]]+$ ]] ;then
    jiecheng $n
else
    echo "请输入一个正整数!"
    exit 2
fi

技术分享

 

示例:

 

1、编写服务脚本/root/bin/testsrv.sh,完成如下要求

(1) 脚本可接受参数:start, stop, restart, status

(2) 如果参数非此四者之一,提示使用格式后报错退出

(3) 如是start:则创建/var/lock/subsys/SCRIPT_NAME, 并显示“启动成功”

考虑:如果事先已经启动过一次,该如何处理?

(4) 如是stop:则删除/var/lock/subsys/SCRIPT_NAME, 并显示“停止完成”

考虑:如果事先已然停止过了,该如何处理?

(5) 如是restart,则先stop, 再start

考虑:如果本来没有start,如何处理?

(6) 如是status, 则如果/var/lock/subsys/SCRIPT_NAME文件存在,则显示“SCRIPT_NAMEis running…”

如果/var/lock/subsys/SCRIPT_NAME文件不存在,则显示“SCRIPT_NAME is stopped…”

其中:SCRIPT_NAME为当前脚本名

#!/bin/bash
fstart () {
    if [ -e "$1"  ]; then
        echo "Already start"
    else
        touch $ff
        echo "Start Successfully!"
    fi
}
 
fstop () {
    if [ -e "$1" ] ; then
        rm -f $1
        echo "Stop Successfully!"
    else
        echo "Already stop"
    fi
}
 
frestart() {
    if [ -e "$1" ] ; then
        fstop &> /dev/null
        fstart &> /dev/null
        echo "Restart successful"
    else
        fstart &> /dev/null
        echo "start successful " 
    fi
}
 
fstatus () {
    if [ -e "$1" ] ;then
        echo "testsrv is runing..."
    else
        echo "testsrv is stopped..."
    fi
}
 
select choice in start stop restart status quit
do
    ff=/var/lock/subsys/testsrv.sh
    case $choice in
    start)
        fstart $ff
        ;;
    stop)
        fstop $ff
        ;;
    restart)
        frestart $ff
        ;;
    status)
        fstatus $ff
        ;;
    quit)
        break
        ;;
    *)
        echo "error!"
        exit 2
    esac
done

技术分享

2、编写脚本/root/bin/copycmd.sh

(1) 提示用户输入一个可执行命令名称;

(2) 获取此命令所依赖到的所有库文件列表

(3) 复制命令至某目标目录(例如/mnt/sysroot)下的对应路径下;

如:/bin/bash ==> /mnt/sysroot/bin/bash

/usr/bin/passwd==> /mnt/sysroot/usr/bin/passwd

(4) 复制此命令依赖到的所有库文件至目标目录下的对应路径下:

如:/lib64/ld-linux-x86-64.so.2 ==> /mnt/sysroot/lib64/ld-linux-x86-64.so.2

(5)每次复制完成一个命令后,不要退出,而是提示用户键入新的要复制的命令,并重复完成上述功能;直到用户输入quit退出

#!/bin/bash
 
fcopy () {
    aa=`which $1 | tail -1`
   ldd $aa 
   if [ $? -eq 0 ] ;then
    cp --parents $aa  /mnt/sysroot/ && echo "Command has been copied ! "
    ldd $aa | sed  -r  -e  "[email protected]*[[:space:]]+(/.*)[[:space:]].*@\[email protected]" -e ‘1d‘ > /testdir/copycmd 
        while read line ;do
            cp --parents $line /mnt/sysroot/ 
        done < /testdir/copycmd
     echo "Libary file has been copied !"
    fi  
}   
 
select choice in  run  quit ;do 
    case $choice in
    run)
        read -p "Please enter a command:" cmd 
        fcopy $cmd
        ;;  
    quit)
        echo "bye"
        exit 0
        ;;  
    *)  
        echo "error"
        exit 2
    esac
done

技术分享

技术分享

3、编写函数实现两个数字做为参数,返回最大值

#!/bin/bash
fmax () {
    if [ $1 -lt $2 ] ;then
        echo "$2"
    else
        echo "$1"
    fi  
}
  
fmax $1 $2

技术分享

4、编写函数实现数字的加减乘除运算,例如输入1 + 2,,将得出正确结果

#!/bin/bash
fjia () {
    sum1=$[$1+$2]
    echo $sum1
}
 
fjian () {
    sum2=$[$1-$2]
    echo $sum2
}
 
fcheng () {
    sum3=$[$1*$2]
    echo $sum3
}
 
fchu () {
    sum4=$[$1/$2]
    echo $sum4
}
read -p "please input: " n j m 
    case $j in
    +)  
        fjia $n $m
        ;;  
    -)  
        fjian $n $m
        ;;  
    \*) 
        fcheng $n $m
        ;;  
    /)
        fchu $n $m
        ;;
    *)
        echo "error"
        exit 2
    esac

技术分享

5、斐波那契数列又称黄金分割数列,因数学家列昂纳多·斐波那契以兔子繁殖为例子而引入,故又称为“兔子数列”,指的是这样一个数列:0、1、1、2、3、5、8、13、21、34、……,斐波纳契数列以如下被以递归的方法定义:F(0)=0,F(1)=1,F(n)=F(n-1)+F(n-2)(n≥2)

写一个函数,求n阶斐波那契数列

#!/bin/bash
fbnq () {
    if [ $1 -eq 0  ];then
        echo 0
    elif [ $1 -eq 1 ];then
        echo 1
    else
        echo "$[$(fbnq $[$1-1])+$(fbnq $[$1-2])]"
    fi  
}
read -p "please enter a number:" n
    fbnq $n

技术分享

6、汉诺塔(又称河内塔)问题是源于印度一个古老传说。大梵天创造世界的时候做了三根金刚石柱子,在一根柱子上从下往上按照大小顺序摞着64片黄金圆盘。大梵天命令婆罗门把圆盘从下面开始按大小顺序重新摆放在另一根柱子上。并且规定,在小圆盘上不能放大圆盘,在三根柱子之间一次只能移动一个圆盘。

利用函数,实现N片盘的汉诺塔的移动步骤

#!/bin/bash
fpan () {
    if [ $1 -eq 1 ];then
        echo "$2 ==>  $4"
    else
        fpan $[$1-1] $2 $4 $3
            echo "$2 ==> $4"
        fpan $[$1-1] $3 $2 $4
    fi  
}
 
fpan $1 a b c

技术分享


以上是关于shell脚本进阶的主要内容,如果未能解决你的问题,请参考以下文章

SHELL脚本编程进阶

Shell脚本之进阶

代码片段:Shell脚本实现重复执行和多进程

shell脚本进阶

shell 脚本 片段

Linux shell脚本进阶使用