zoukankan      html  css  js  c++  java
  • shell流程控制(流程不可为空、if else流程、for循环、while循环/无限循环、until循环、分支结构case...esac、跳出循环)、shell 函数(定义需在开头、如何调用、有无return返回值差别、函数参数$1 / ${10}、$? 获取返回值及其限制)、shell 输入输出重定向、Here Document重定向方式、/dev/null文件作用、shell文件包含及使用

    一、shell流程控制

    1、和其他语言不一样,sh 的流程控制不可为空。如果 else 分支没有语句执行,就不要写这个 else。

    2、if else 流程

    (1)if 语句语法格式:

    if condition
    then
        command1 
        command2
        ...
        commandN 
    fi

    写成一行(适用于终端命令提示符):

    if [ $(ps -ef | grep -c "ssh") -gt 1 ]; then echo "true"; fi

    末尾的 fi 就是 if 倒过来拼写,表示 if 结束。

    (2)if else 语法格式:

    if condition
    then
        command1 
        command2
        ...
        commandN
    else
        command
    fi

    (3)if else-if else 语法格式:

    if condition1
    then
        command1
    elif condition2 
    then 
        command2
    else
        commandN
    fi
    #以下实例判断两个变量是否相等:
    a=10
    b=20
    if [ $a == $b ]
    then
       echo "a 等于 b"
    elif [ $a -gt $b ]
    then
       echo "a 大于 b"
    elif [ $a -lt $b ]
    then
       echo "a 小于 b"
    else
       echo "没有符合的条件"
    fi
    
    #输出结果:a 小于 b

    3、for 循环   ——   for循环一般格式为:

    for var in item1 item2 ... itemN
    do
        command1
        command2
        ...
        commandN
    done

      写成一行:

    for var in item1 item2 ... itemN; do command1; command2 done;

      当变量值在列表里,for 循环即执行一次所有命令,使用变量名获取列表中的当前取值。命令可为任何有效的 shell 命令和语句。in 列表可以包含替换、字符串和文件名。

      in列表是可选的,如果不用它,for循环使用命令行的位置参数。

    #例如,顺序输出当前列表中的数字:
    for loop in 1 2 3 4 5
    do
        echo "The value is: $loop"
    done
    
    #顺序输出字符串中的字符:
    #!/bin/bash
    for str in This is a string
    do
        echo $str
    done

    4、while 循环  ——  用于不断执行一系列命令,也用于从输入文件中读取数据。其语法格式为:

    while condition
    do
        command
    done
    #while循环可用于读取键盘信息。下面的例子中,输入信息被设置为变量FILM,按<Ctrl-D>结束循环。
    echo '按下 <CTRL-D> 退出'
    echo -n '输入你最喜欢的网站名: '
    while read FILM
    do
        echo "是的!$FILM 是一个好网站"
    done

    5、until 循环   ——   until 循环执行一系列命令直至条件为 true 时停止。until 循环与 while 循环在处理方式上刚好相反。一般 while 循环优于 until 循环,但在某些时候—也只是极少数情况下,until 循环更加有用。

      until 语法格式:condition 一般为条件表达式,如果返回值为 false,则继续执行循环体内的语句,否则跳出循环。

    until condition
    do
        command
    done

    5、分支结构 case ... esac

    (1)case ... esac 为多选择语句,与其他语言中的 switch ... case 语句类似,是一种多分枝选择结构,每个 case 分支用右圆括号开始,用两个分号 ;; 表示 break,即执行结束,跳出整个 case ... esac 语句,esac(就是 case 反过来)作为结束标记。

    (2)可以用 case 语句匹配一个值与一个模式,如果匹配成功,执行相匹配的命令。case ... esac 语法格式如下:

    case  in
    模式1)
        command1
        command2
        ...
        commandN
        ;;
    模式2
        command1
        command2
        ...
        commandN
        ;;
    esac

      case 工作方式如上所示,取值后面必须为单词 in,每一模式必须以右括号结束。取值可以为变量或常数,匹配发现取值符合某一模式后,其间所有命令开始执行直至 ;;。

    (3)取值将检测匹配的每一个模式。一旦模式匹配,则执行完匹配模式相应命令后不再继续其他模式。如果无一匹配模式,使用星号 * 捕获该值,再执行后面的命令。

    #下面的脚本提示输入 1 到 2,与每一种模式进行匹配:
    echo '输入 1 到 4 之间的数字:'
    echo '你输入的数字为:'
    read aNum
    case $aNum in
        1)  echo '你选择了 1'
        ;;
        2)  echo '你选择了 2'
        ;;
        *)  echo '你没有输入 1 到 2 之间的数字'
        ;;
    esac

    6、跳出循环  ——  在循环过程中,有时候需要在未达到循环结束条件时强制跳出循环,Shell使用两个命令来实现该功能:break和continue。

    (1)break命令允许跳出所有循环(终止执行后面的所有循环)。

    (2)continue命令与break命令类似,只有一点差别,它不会跳出所有循环,仅仅跳出当前循环。

    二、shell 函数

    1、linux shell 可以用户定义函数,然后在shell脚本中可以随便调用。shell中函数的定义格式如下:

    [ function ] funname [()]
    {
        action;
        [return int;]
    }

      说明:(1)可以带function fun() 定义,也可以直接fun() 定义,不带任何参数。(2)参数返回,可以显示加:return 返回,如果不加,将以最后一条命令运行结果,作为返回值。 return后跟数值n(0-255)

    # 1、定义了一个函数并进行调用,没有return,使用最后一行语句运行结果的返回值
    demoFun(){
        echo "这是我的第一个 shell 函数!"
    }
    demoFun
    # 这是我的第一个 shell 函数!
    
    # 2、定义一个带有return语句的函数,使用 return 的结果作为返回值
    funWithReturn(){
        echo "输入第一个数字: "
        read aNum
        echo "输入第二个数字: "
        read anotherNum
        echo "两个数字分别为 $aNum 和 $anotherNum !"
        return $(($aNum+$anotherNum))
    }
    funWithReturn
    echo "输入的两个数字之和为 $? !"

      函数返回值在调用该函数后通过 $? 来获得

      注意:(1)所有函数在使用前必须定义。这意味着必须将函数放在脚本开始部分,直至shell解释器首次发现它时,才可以使用。(2)调用函数仅使用其函数名即可。

    2、函数参数:

    (1)在Shell中,调用函数时可以向其传递参数。在函数体内部,通过 $n 的形式来获取参数的值,例如,$1表示第一个参数,$2表示第二个参数

    funWithParam(){
        echo "第一个参数为 $1 !"
        echo "第二个参数为 $2 !"
        echo "第十个参数为 $10 !"
        echo "第十个参数为 ${10} !"
        echo "第十一个参数为 ${11} !"
        echo "参数总数有 $# 个!"
        echo "作为一个字符串输出所有参数 $* !"
    }
    funWithParam 1 2 3 4 5 6 7 8 9 34 73
    
    #结果
    第一个参数为 1 !
    第二个参数为 2 !
    第十个参数为 10 !
    第十个参数为 34 !
    第十一个参数为 73 !
    参数总数有 11 个!
    作为一个字符串输出所有参数 1 2 3 4 5 6 7 8 9 34 73 !

    (2)注意,$10 不能获取第十个参数,获取第十个参数需要${10}。当n>=10时,需要使用${n}来获取参数

    3、$? 仅对其上一条指令负责,一旦函数返回后其返回值没有立即保存入参数,那么其返回值将不再能通过 $? 获得。比如:

    demoFun2
    echo $?
    demoFun1
    echo 在这里插入命令!
    echo $?

      调用 demoFun2 后,函数最后一条命令 expr 1 + 1 得到的返回值($?值)为 0,意思是这个命令没有出错(需要注意的是:没有使用 return,其返回值就不是 2 哦)。所有的命令的返回值仅表示其是否出错,而不会有其他有含义的结果

      第二次调用 demoFun1 后,没有立即查看 $? 的值,而是先插入了一条别的 echo 命令,最后再查看 $? 的值得到的是 0,也就是上一条 echo 命令的结果,而 demoFun1 的返回值被覆盖了。

      下面这个测试,连续使用两次 echo $?,得到的结果不同,更为直观:

    #!/bin/bash
    function demoFun1(){
        echo "这是我的第一个 shell 函数!"
        return `expr 1 + 1`
    }
    
    demoFun1
    echo $?  # 显示的是 函数 demoFun1 的返回值 2
    echo $?  # 显示的是 语句 echo $? 的返回值 0 ,代表运行没有报错
    
    #输出结果:
    这是我的第一个 shell 函数!
    2
    0

    4、函数与命令的执行结果可以作为条件语句使用。要注意的是:shell 语言中 0 代表 true,0 以外的值代表 false

    三、shell 输入/输出重定向

    1、大多数 UNIX 系统命令从你的终端接受输入并将所产生的输出发送回​​到您的终端。一个命令通常从一个叫标准输入的地方读取输入,默认情况下,这恰好是你的终端。同样,一个命令通常将其输出写入到标准输出,默认情况下,这也是你的终端。

      重定向命令列表如下:

    命令说明
    command > file 将输出重定向到 file。
    command < file 将输入重定向到 file。
    command >> file 将输出以追加的方式重定向到 file。
    n > file 将文件描述符为 n 的文件重定向到 file。
    n >> file 将文件描述符为 n 的文件以追加的方式重定向到 file。
    n >& m 将输出文件 m 和 n 合并。
    n <& m 将输入文件 m 和 n 合并。
    << tag 将开始标记 tag 和结束标记 tag 之间的内容作为输入。

      需要注意的是文件描述符 0 通常是标准输入(STDIN),1 是标准输出(STDOUT),2 是标准错误输出(STDERR)。

    2、输出重定向command1 > file1 (执行command1然后将输出的内容存入file1)

      注意:任何file1内的已经存在的内容将被新内容替代。如果要将新内容添加在文件末尾,请使用>>操作符。

    3、输入重定向command1 < file1 (本来需要从键盘获取输入的命令会转移到文件读取内容)

    4、重定向深入讲解:

      一般情况下,每个 Unix/Linux 命令运行时都会打开三个文件:

    标准输入文件(stdin):stdin的文件描述符为0,Unix程序默认从stdin读取数据。

    标准输出文件(stdout):stdout 的文件描述符为1,Unix程序默认向stdout输出数据。

    标准错误文件(stderr):stderr的文件描述符为2,Unix程序会向stderr流中写入错误信息。

      默认情况下,command > file 将 stdout 重定向到 file,command < file 将stdin 重定向到 file。

      如果希望 stderr 重定向到 file,可以这样写:command 2>file  (stderr 追加到 file 文件末尾,可以这样写:command 2>>file)

      如果希望将 stdout 和 stderr 合并后重定向到 file,可以这样写:

    $ command > file 2>&1
    #或者
    $ command >> file 2>&1

    5、Here Document 是 Shell 中的一种特殊的重定向方式,用来将输入重定向到一个交互式 Shell 脚本或程序。它的基本的形式如下:

    command << delimiter
        document
    delimiter

      它的作用是将两个 delimiter 之间的内容(document) 作为输入传递给 command

      注意:(1)结尾的delimiter 一定要顶格写,前面不能有任何字符,后面也不能有任何字符,包括空格和 tab 缩进。(2)开始的delimiter前后的空格会被忽略掉。

      这个我们在项目里制作 postgresql 镜像运行初始化脚本的时候有用到,当时不懂,现在了解了原来就是这个。

    # 在命令行中通过 wc -l 命令计算 Here Document 的行数:
    $ wc -l << EOF
        欢迎来到
        www.runoob.com
    EOF
    2          # 输出结果为 2 行
    
    # 将 Here Document 用在脚本中
    cat << EOF
    欢迎来到
    www.runoob.com
    EOF
    
    #输出结果:
    欢迎来到
    www.runoob.com

    6、/dev/null 文件  ——   command > /dev/null

    (1)如果希望执行某个命令,但又不希望在屏幕上显示输出结果,那么可以将输出重定向到 /dev/null

      /dev/null 是一个特殊的文件,写入到它的内容都会被丢弃;如果尝试从该文件读取内容,那么什么也读不到。但是 /dev/null 文件非常有用,将命令的输出重定向到它,会起到"禁止输出"的效果

    (2)如果希望屏蔽 stdout 和 stderr,可以这样写:

    $ command > /dev/null 2>&1

    注意:0 是标准输入(STDIN),1 是标准输出(STDOUT),2 是标准错误输出(STDERR)。

    这里的 2 和 > 之间不可以有空格,2> 是一体的时候才表示错误输出

    四、文件包含

    1、Shell 也可以包含外部脚本。这样可以很方便的封装一些公用的代码作为一个独立的文件。Shell 文件包含的语法格式如下:

    . filename   # 注意点号(.)和文件名中间有一空格
    #
    source filename

    2、实例展示

    # test1.sh 代码如下:
    
    #!/bin/bash
    url="http://www.runoob.com"
    
    3 test2.sh 代码如下:
    
    #!/bin/bash
    #使用 . 号来引用test1.sh 文件
    . ./test1.sh
    
    # 或者使用以下包含文件代码
    # source ./test1.sh
    
    echo "官网地址:$url"

      接下来,我们为 test2.sh 添加可执行权限并执行:

    $ chmod +x test2.sh
    $ ./test2.sh
    
    #官网地址:http://www.runoob.com

      提示:被包含的文件 test1.sh 不需要可执行权限。

  • 相关阅读:
    Matlab 绘制三维立体图(以地质异常体为例)
    Azure DevOps的variable group实现array和hashtable参数的传递
    Azure DevOps 利用rest api设置variable group
    Azure AADSTS7000215 其中一种问题的解决
    Power BI 实现实时更新Streaming Dataset
    AAD Service Principal获取azure user list (Microsoft Graph API)
    Matlab 沿三维任意方向切割CT图的仿真计算
    Azure Powershell script检测登陆并部署ARM Template
    Azure KeyVault设置策略和自动化添加secrets键值对
    Azure登陆的两种常见方式(user 和 service principal登陆)
  • 原文地址:https://www.cnblogs.com/goloving/p/15170862.html
Copyright © 2011-2022 走看看