bash函数定义/使用/传参…

函数:function, 功能
    过程式编程,代码重用
        模块化编程
        简洁
       
    语法:
        function f_name {
            函数体
        }
        或者
        f_name() {
            函数体
        }
   
    调用:使用函数名
        函数名出现的地方,会被自动替换为函数代码;
       
    练习:利用函数改写此前的服务脚本
        启动一个脚本之后,脚本本身会在/var/lock/subsys/目录下创建一个对应的锁文件
        service httpd start启动httpd服务,此时在/var/lock/subsys/目录下,会生成httpd锁文件
        /etc/rc.d/init.d存放服务脚本,遵循的是lsb编写格式
        在/etc/rc.d/init.d中有一个文件functions,为公用的函数
        想要调用functions文件里的各个函数,则添加
        . /etc/rc.d/init.d/functions代码,在你需要调用的文件中
        假如是httpd服务,则需要另外导入 . /etc/sysconfig/httpd 的配置文件
        #!/bin/bash
        #
        #$0是当前脚本的名称
        prog=`basename $0`
        lockfile=/var/lock/subsys/$prog
       
        start() {
            if [ -e $lockfile ];then
                echo "$prog is already running."
            else
                touch $lockfile
                [ $? -eq 0 ] && echo "Starting $prog finished."
            fi
        }
       
        stop() {
            if [ -e $lockfile ];then
                rm -rf $lockfile
                [ $? -eq 0 ] && echo "Stoping $prog finished."
            else
                echo "$prog is stopped yet."
            fi
        }
       
        case $1 in
        "start")
            start ;;
        "stop")
            stop ;;
        "restart")
            stop
            start ;;
        *)
            echo "Usage: $prog {start|stop|restart}"
            exit 1
        esac
       
    函数返回值:
        函数的执行结果返回值:
            函数中使用打印语句:echo, printf
            函数体中OS命令执行结果的输出
       
        函数的退出状态码:
            默认取决于函数体执行的最后一个命令的退出状态码;
            自定义退出状态码:
                return [0-255]
                注意:函数体运行时,一旦遇到return语句,函数即返回;
       
        函数可接受参数:
            传递参数给函数,调用函数时,在函数名后给出参数列表即可;
                例: testfunc arg1 arg2 arg3
            在函数体中可使用$1,$2....来调用传递过来的各参数
            可使用类似脚本的特殊变量
                $*,$@: 一次性获取参数列表
                $#: 参数的个数
                $?: 上一条命令的执行状态或返回状态码
            例:
            #!/bin/bash
            #
            showuserinfo() {
                [ $# -lt 1 ] && return 1 #参数个数小于1,即没有传递参数

                ! id $1 &> /dev/null && return 2 #用户名不存在

                grep "^$1\>" /etc/passwd | cut -d: -f7
                [ $? -eq 0 ] && return 0 || return 3 #前一条命令执行成功,则return 0, 否则return 3
            }

            while true; do
                read -p "Enter a username: " username
                [ "$username" == 'quit' ] && break
                showuserinfo $username
                [ $? -ne 0 ] && echo "There is something wrong."
            done       
       
    练习:写一个脚本,完成如下功能(使用函数):
    1、提示用户输入一个可执行命令;
    2、获取这个命令所依赖的所有库文件(使用ldd命令);
    3、复制命令至/mnt/sysroot/对应的目录中
        解释:
            假设,如果复制的是cat命令,其可执行程序的路径是/bin/cat,
                那么就要将/bin/cat复制到/mnt/sysroot/bin/目录中,
            如果复制的是useradd命令,而useradd的可执行文件路径为/usr/sbin/useradd,
                那么就要将其复制到/mnt/sysroot/usr/sbin/目录中;
    4、复制各库文件至/mnt/sysroot/对应的目录中;
       
    #!/bin/bash
    #
    target=/mnt/sysroot/
    preCmd(){
        #which $1:获取命令的所在目录, 例如which cat
        #which --skip-alias $1: 参数--skip-alias是跳过别名解析,例如which cp和which --skip-alias cp
        if which $1 &> /dev/null; then
            cmdpath=$(which --skip-alias $1)
            return 0
        else
            echo "No such command."
            return 1
        fi
    }
   
    #/bin/cat
    #/mnt/sysroot/bin/cat
    cmdCopy(){
        #dirname $cmdpath: 获取后边参数所在的目录
        cmddir=$(dirname $cmdpath)
        [ -d $target/$cmddir ] || mkdir -p $target/$cmddir
        [ -f $target/$cmdpath ] || cp $target/$cmddir
        return 0
    }
   
    #ldd /bin/cat: 查看cat命令所需的库文件
    libCopy(){
        for lib in $(ldd $1 | grep -E -o "/[^[space::]]+");do
            libdir=$(dirname $lib)           
            [ -d $target/$libdir ] || mkdir -p $target/$libdir
            [ -f $target/$lib ] || cp $target/$libdir
        done
        return 0
    }
   
    main(){
        while true; do
            read -p "Plz enter a command(quit):" cmd
            [ "$cmd" == 'quit' ] && break
            precmd $cmd
            if [ $? -eq 0 ];then
                cmdCopy $cmdpath
                libCopy $cmdpath
            fi
        done
    }
    main
   
    变量的作用域:
        本地变量:整个脚本,在脚本的函数中也可调用,也可修改;
        局部变量:函数调用的整个生命周期;
   
    函数递归:
        函数直接或间接调用函数自身
           
            1 1 2 3 5 8 13
            阶乘: 10!
                n(n-1)!
                n(n-1)(n-2)!
       
                阶乘函数:
                fact() {
                    if [ $1 -eq 0 -o $1 -eq 1 ];then
                        echo 1
                    else
                        echo $[$1*$(fact $[$1-1])]
                    fi
                }

posted @ 2015-10-30 18:30  lucky_zhang  阅读(5471)  评论(0编辑  收藏  举报