zoukankan      html  css  js  c++  java
  • 菜鸟教程之学习Shell script笔记(下)

    菜鸟教程Shell script学习笔记(下)

    以下内容是学习菜鸟教程之shell教程,所整理的笔记

    菜鸟教程之shell教程:http://www.runoob.com/linux/linux-shell.html

    Shell流程控制

    和Java、PHP等语言不一样,sh的流程控制不可为空,如(以下为PHP流程控制写法):

    <?php
    if (isset($_GET["q"])){
        search(q);
    }
    else {
        //不做任何事情
    }
    #在sh/bash里不可以这么写,如果else分支没有语句执行,就不要写这个else.
    
    • if else
    if 语句语法格式:
    if condition
    then
        command1
        command2
        ...
    fi
    #写成一行(适用于终端)
    
    if else 语法格式:
    if condition
    then
    command1
    command2
    ...
    else
        command
    fi
    
    if else-if else语法格式:
    if condition1
    then
        command1
    elif condition2
    then
        command2
    else
        commandN
    fi
    #if else语句经常与test命令结合适用,如下所示:
    #!/bin/bash
    num1=$[2*3]
    num2=$[1+5]
    if test $[num1] -eq $[num2]
    then
        echo '两个数字相等!'
    else
        echo '两个数字不相等!'
    fi
    
    • for 循环
    for var in item1 item2 ... itemN
    do
        command1
        command2
        ...
        command3
    done
    也可以写成一行:for var in item1 item2 ... itemN; do ocmmand1; command2... done;
    注意:
    当变量值在列表里,for循环即执行一次所有命令,使用变量名获取列表中的当前取值。命令可为任何有效的shell命令和语句。in列表可以包含替换、字符串和文件名。
    in列表是可选的,如果不用它,for循环使用命令行的位置参数。
    例如,顺序输出当前列表中的数字:
    for loop in 1 2 3 4 5
    do
        echo "The value is: $loop"
    done
    #如果加上'',in则不是列表,而是变成了字符串,列表是一个个字符输出。
    
    • while 语句:
      while循环用于不断执行一系列命令,也用于从输入文件中读取数据;命令通常为测试条件。其格式为:
    while condition
    do
        command
    done
    
    while 循环可以用于读取键盘信息:
    #!/bin/bash
    echo '按下<CTRL-D>退出'
    echo -n '输入你最喜欢的网站名:'
    while read FILM
    do
        echo "是的!$FILM是一个好网站"
    done
    
    • 无限循环
    while:
    do
        command
    done
    或者:
    while true
    do
        command
    done
    或者:
    for (( ; ; ))
    
    • until 循环
    until 循环执行一系列命令直至条件为 true 时停止。
    until 循环与 while 循环在处理方式上刚好相反。
    一般 while 循环优于 until 循环,但在某些时候—也只是极少数情况下,until 循环更加有用。
    until 语法格式:
    until condition
    do
        command
    done
    condition 一般为条件表达式,如果返回值为 false,则继续执行循环体内的语句,否则跳出循环。
    
    • case
      Shell case语句为多选择语句。可以用case语句匹配一个值与一个模式,如果匹配成功,执行相匹配的命令。case语句格式如下:
    case 值 in
    模式1)
        command1
        command2
        ...
        commandN
        ;;
    模式2)
        command1
        command2
        ...
        ;;
    esac
    case工作方式如上所示。取值后面必须为单词in,每一模式必须以右括号结束。取值可以为变量或常数。匹配发现取值符合某一模式后,其间所有命令开始执行直至 ;;。
    取值将检测匹配的每一个模式。一旦模式匹配,则执行完匹配模式相应命令后不再继续其他模式。如果无一匹配模式,使用星号 * 捕获该值,再执行后面的命令。
    eg:
    echo '输入1到4之间的数字:'
    echo '你输入的数字为:'
    read aNum
    case $aNum in
        1) echo '你选择了 1'
        ;;
        2) echo '你选择了 2'
        ;;
        3) echo '你选择了 3'
        ;;
        4) echo '你选择了 4'
        ;;
        *) echo '你没有输入1到4之间的数字'
        ;;
    esac
    
    • 跳出循环
      使用break和continue强制跳出循环
    • break命令
      break命令允许跳出所有循环(终止执行后面的所有循环)
    • continue
      continue命令与break命令类似,只有一点差别,它不会跳出所有循环,仅仅跳出当前循环。
    eg:
    #!/bin/bash
    while:
    do
        echo -n "输入1到5之间的数字:"
        read aNum
        case $aNum in
            1|2|3|4|5) echo "你输入的数字为 $aNum!"
            ;;
            *) echo "你输入的数字不是1到5之间的!"
                continue
                echo "游戏结束"
            ;;
        esac
    done
    运行代码发现,当输入大于5的数字时,该例中的循环不会结束,语句 echo "游戏结束" 永远不会被执行。
    而如果在break后面循环体内,加上一句echo 语句也不被执行!
    
    • esac
      case的语法和C family语言差别很大,它需要一个esac(就是case反过来)作为结束标记,每个case分支用右圆括号,用两个分号表示break。

    Shell函数

    linux shell 可以用户定义函数,然后在shell脚本中可以随便调用。

    shell中函数的定义格式如下:

    [ function ] funname [()]
    {
        action;
        [return int;]
    }
    注意:
    1、可以带function fun() 定义,也可以直接fun() 定义,不带任何参数。
    2、参数返回,可以显示加:return 返回,如果不加,将以最后一条命令运行结果,作为返回值。 return后跟数值n(0-255
    下面的例子定义了一个函数并进行调用:
    #!/bin/bash
    demoFun(){
        echo "这是我的第一个Shell函数!"
    }
    echo "-----函数开始执行-----"
    demoFun
    echo "-----函数执行完毕-----"
    
    下面定义一个带return语句的函数:
    #!/bin/bash
    fuunWithReturn(){
        echo "这个函数会对输入的两个数字进行想加运算..."
        echo "输入第一个数字:"
        read aNum
        echo "输入第二个数字:"
        read anotheerNum
        echo "两个数字分别为$aNum 和$anotherNum!"
        return $(($aNum+$anotherNum))
    }
    funWithReturn
    echo "输入的两个数字之和为 $? !"
    注意:
    函数返回值在调用该函数后通过 $? 来获得。
    所有函数在使用前必须定义。这意味着必须将函数放在脚本开始部分,直至shell解释器首次发现它时,才可以使用。调用函数仅使用其函数名即可。
    
    • 函数参数
      在Shell中,调用函数时可以向其传递参数。在函数体内部,通过 $n 的形式来获取参数的值,例如,$1表示第一个参数,$2表示第二个参数... 当n>10的时候,需要使用${n}来获取参数
    $# 传递到脚本的参数个数
    $* 以一个单字符穿显示所有向脚本传递的参数
    $$ 脚本运行的当前进程ID号
    $! 脚本运行的最后一个进程ID号
    $@ 与$*相同,但是使用时要加引号,并在引号中返回每个参数
    $- 显示shell使用的当前选项,与set命令功能相同
    $? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误
    eg:
    #!/bin/bash
    funWithParam(){
        echo "第一个参数为 $1 !"
        echo "第二个参数为 $2 !"
        echo "第十个参数为 $10 !"
        echo "第十个参数为 ${10} !"
        echo "第十一个参数为 ${11} !"
        echo "参数总数有 $# 个!"
        echo "作为一个字符串输出所有的参数 $* !"
    }
    runWithParam 1 2 3 4 5 6 7 8 9 34 73
    

    Shell 输入/输出重定向

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

    重定向命令列表如下:

    command > file 将输出重定向到file
    command < file 将输入重定向到file
    command >> file 将输出以追加的方式重定向到file
    n > file 将文件描述符为n的文件以追加的方式重定向到file
    n >& m 将输出文件m and n合并
    n <& m 将输入文件m and n合并
    << tag 将开始标记tag和结束标记tag之间的内容作为输入
    注意:
    需要注意的是文件描述符 0 通常是标准输入(STDIN),1 是标准输出(STDOUT),2 是标准错误输出(STDERR)。
    
    • 输出重定向
    输出重定向详细:
    mkdir users
    who > users
    cat users
    输出重定向会覆盖文件内容,请看下面例子:
    echo "www.baidu.com" > users
    cat users
    www.baidu.com #只有这一行内容了
    如果不希望文件内容被覆盖,可以使用>>追加到文件末尾,例如:
    echo "www.google.com" >> users
    cat users
    www.baidu.com
    www.google.com
    
    • 输入重定向
    command < file1
    本来需要从键盘获取输入的命令会转移到文件读取内容
    注意:输出重定向是大于号(>),输入重定向是小于号(<)。
    eg:
    wc -l users #output:2 users
    wc -l < users #output:2
    注意:上面两个例子的结果不同:第一个例子,会输出文件名;第二个不会,因为它仅仅知道从标准输入读取内容。
    command1 < infile > outfile
    同时替换输入和输出,执行command1,从文件infile读取内容,然后将输出写入到outfile中。
    
    • 重定向深入讲解
    一般情况下,每个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
    command 2 >> file # stderr追加到file文件末尾
    command > file 2>&1 # 将stdout 和 stderr合并后重定向到file
    command < file1 > file2 #对stdin 和 stdout 都重定向,command 命令将 stdin 重定向到 file1,将 stdout 重定向到 file2。
    
    • Here Document
      Here Document 是 Shell 中的一种特殊的重定向方式,用来将输入重定向到一个交互式 Shell 脚本或程序。它的基本的形式如下:
    command << delimiter
        document
    delimiter
    它的作用是将两个delimiter之间的内容(document)作为输入传递给command.
    注意:
    结尾的delimiter 一定要顶格写,前面不能有任何字符,后面也不能有任何字符,包括空格和 tab 缩进。
    开始的delimiter前后的空格会被忽略掉。
    eg:
    在命令行中通过wc-l 命令计算Here Doucument 的行数:
    wc -l << EOF
        test
        test!
        www.google.com
    EOF
    #output:3 EOF就是delimiter的一种写法
    也可以将Here Document用在脚本中:
    #!/bin/bash
    cat << EOF
    欢迎来到
    菜鸟教程
    www.ruunoob.com
    EOF
    
    • /dev/null 文件
      如果希望执行某个命令,但又不希望在屏幕上显示输出结果,那么可以将输出重定向到 /dev/null:
    command > /dev/null
    /dev/null是一个特殊的文件,写入到它的内容都会被丢弃;如果尝试从该文件读取内容,那么什么也读不到。但是 /dev/null 文件非常有用,将命令的输出重定向到它,会起到"禁止输出"的效果。
    如果希望屏蔽 stdout 和 stderr,可以这样写:
    command > /dev/null 2>&1
    注意:0 是标准输入(STDIN),1 是标准输出(STDOUT),2 是标准错误输出(STDERR)。
    

    Shell文件包含

    和其他语言一样,Shell 也可以包含外部脚本。这样可以很方便的封装一些公用的代码作为一个独立的文件。

    Shell 文件包含的语法格式如下:

    . filename #注意点好(.)和文件名中间有一空格
    or
    source filename
    eg:
    创建两个shell脚本文件
    test1.sh代码如下:
    #!/bin/bash
    url="http://www.runoob.com"
    test1.sh代码如下:
    #!/bin/bash
    . ./test1.sh
    echo "菜鸟教程官网地址:$url"
    

    Shell script学习总结

    在菜鸟教程上面学习的Shell script是基础的。如何组合利用,还需要进一步学习。

  • 相关阅读:
    Redis安装配置
    Git本地服务器搭建
    JDK安装配置
    ssh免密登录
    设计模式
    IDEA 快捷键
    LeetCode Sliding Window Maximum
    ElasticSearch 使用小结
    LeetCode Product of Array Except Self
    LeetCode Delete Node in a Linked List
  • 原文地址:https://www.cnblogs.com/hudunkey/p/10281246.html
Copyright © 2011-2022 走看看