• 【Shell编程】Shell基本语法


    Shell 语法

      Shell程序设计作为一种脚本语言,在Linux系统中有广泛的应用,本文记录了关于Shell程序设计的基础语法知识和常用命令,方便查询,熟练使用shell也需要经常实践,这对于完成一些较简单的编程任务很有帮助。


    (1)变量

      在shell里,使用变量之前并不需要事先做出声明,可以通过使用直接创建。默认情况下,所有的变量都被当做字符串进行存储,变量名区分大小写。变量名之前加一个$符号可以访问它的内容。

      输入和输出:可以使用echo命令将一个变量的内容输出到终端,使用read命令(停下来等待用户输入)可以将用户的输入赋值给一个变量。

      单引号和双引号:shell脚本的文件参数以空白符进行分割,如果一个参数包含空白符,就需要放到引号中(防止被解读为不同的参数)。双引号通常可以用来做字符串定界符。而像$salutation这样的变量放到引号中,有这几种情况:不加引号引用变量的值,放到双引号中也引用变量的值,放到单引号中就不引用,而是看作一个字符串。

    $ salutation=Hello
    $ echo $salutation  # 输出 Hello
    $ salutation="Yes Sir"
    $ echo $salutation  # 输出 Yes Sir
    $ salutation=7+2
    $ echo $salutation  # 输出 7+2 (作为字符串看待)
    # 使用引号:
    $ myvar="Hello World"
    $ echo $myvar  # 输出 Hello World
    $ echo "$myvar" # 输出 Hello World
    $ echo '$myvar' # 输出 $myvar
    $ echo $myvar  # 转义字符,输出 $myvar
    

      除了自定义变量外,shell还有一些环境变量和参数变量。

      所谓环境变量,就是指脚本开始执行时,一些变量会根据当前环境设置中的值进行初始化。使用export命令可以设置环境变量。下面的示例展示了常用的环境变量:

    $ echo $HOME # 当前用户的home目录,在我的机器上输出 /home/gzshan
    $ echo $PATH # 以冒号分割的用来搜索命令的目录列表
    $ echo $0 # shell脚本的名字
    $ echo $# # 传递给脚本的参数数量,默认是0
    $ echo $$ # shell脚本的进程号
    

      所谓参数变量,就是指执行脚本程序如果带有参数,一些额外的变量会被创建,如下所示:

    如:运行一个脚本的命令:./first.sh foo bar baz
    $1 $2 $3 #分别指的就是参数foo,bar和baz
    $# # 此时为3
    $@ # 用于列出所有的参数变量,($* 也有相同功能,但受空白符影响,一般用$@)
    

    (2)条件

      条件判断是程序设计语言控制结构的基础,程序需要对条件进行测试和判断,从而执行不同的命令,完成不同的功能和任务。在shell脚本中,完成条件测试有两个命令:test 和 布尔判断命令 [ ]

    • test 命令

        以下的例子展示了test命令的用法,注意如果then和if放在同一行则需要一个分号。

      if test -f fred.c # 如果then写在这一行需要一个分号,这条语句检查一个文件是否存在
      then
      	echo "Hello"
      fi
      
    • 布尔判断命令[ ]

        把 [ 当作一条命令看起来有些奇怪,但是会使得程序变得简单。注意:使用 [ 时,后面必须有空格,还应该使用 ] 来结尾。

      if [ -f fred.c ] ; then # 同样必须有分号,必须有空格
      echo "test"
      fi
      

        当使用以上两个命令时,可以使用的条件类型归结为三类,用下表归纳。

    第一类:字符串的比较 比较结果
    string1 = string2 两个字符串相同结果为真
    string1 != string2 两个字符串不相同结果为真
    -n string 字符串不为空则结果为真
    -z string 字符串为null则结果为真
    第二类:算术比较 比较结果
    expression1 -eq expression2 两个表达式结果相等为真
    expression1 -nq expression2 两个表达式结果不相等为真
    expression1 -gt expression2 expression1 大于 expression2 为真
    expression1 -ge expression2 expression1 大于等于 expression2 为真
    expression1 -lt expression2 expression1 小于 expression2 为真
    expression1 -le expression2 expression1 小于等于 expression2 为真
    !expression expression为假则结果为真
    第三类:与文件有关的条件测试 比较结果
    -d file 文件是一个目录,则结果为真
    -f file 文件是一个普通文件,则结果为真
    -g file 文件set-group-id 被设置,则结果为真
    -u file 文件set-user-id 被设置,则结果为真
    -s file 文件大小不为0,则结果为真
    -r / -w / -x file 文件可读 / 可写 / 可执行,则结果为真
    -a file 文件存在,则结果为为真
    -c file 文件存在并且为字符特殊文件,则结果为真

    (3)控制结构

    • if 语句和elif 语句

        if 语句比较简单,下面的例子展示了if 语句和elif语句的用法。

      #!/bin/sh
      echo "Is it morning? Please answer yes or no"
      read timeofday
      if [ $timeofday = "yes" ] # if语句,后面跟条件测试
      then
      	echo "Good morning"
      elif [ $timeofday = "no" ]; then # then放同一行,加分号
      	echo "Good afternoon"
      else
      	echo "Sorry,$timeofday not recognized. Enter yes or no"
      	exit 1
      fi
      exit 0
      
    • for 语句

        for语句用以循环处理一组值,这组值可以是任意字符串的集合,也可以是其他命令的输出结果,下面用两个例子展示其用法。

      #!/bin/sh
      for foo in bar fud 43 # 循环处理一组字符串
      do
      	echo $foo
      done
      exit 0
      
      #!/bin/sh
      for file in $(ls f*.sh); do # 使用通配符扩展for循环,列出所有以f开头,扩展名为.sh的脚本文件
      	# 说明:$()是执行该命令得到的输出结果
      	lpr $file # lpr是打印命令
      done
      exit 0
      
    • while 语句

        如果事先不知道循环次数,for循环不太好使用的情况下,可以使用while循环。whie语句的do和done之间的语句反复执行,直到条件不再真为止。

      #!/bin/sh
      echo "Enter password"
      read trythis
      while [ $trythis != "secret" ]; do # 反复执行,直到条件不再真为止
      	echo "sorry,try again"
      	read trythis
      done
      exit 0
      
    • until 语句

        until语句与while循环类似,所不同的是,until反复执行循环直到条件为真。

      #!/bin/sh
      until who | grep "$1" > /dev/null
      do
      	sleep 60
      done
      echo -e 'a'
      echo "$1 has just logged in"
      exit 0
      
    • case 语句

        case语句相比其他结构较为复杂,用下面的例子来介绍他的用法,需要特别注意的是:case按顺序查找第一个匹配的模式,而不是最佳匹配。

      #!/bin/sh
      echo "Is it morning? Please answer yes or no"
      read timeofday
      case "$timeofday" in
      	yes | y | Yes | YES )   echo "good morning";; # 注意每个模式末尾是两个分号
      	n* | N* ) echo "good afternoon";;
      	* ) echo "sorry,answer not recognized";;
      esac
      exit 0
      
    • 命令列表

        有时需要将多个命令连接成一个序列,shell提供了命令列表,也就是and列表和or列表,类似于其他程序设计语言,它们也采用的是短路求值。

      if [ -f file1 ] && echo "hello" && [ -f file2 ] && echo "here";then
      	echo "in if"
      fi
      
      if [ -f file ] || echo "hello" || echo "here" ; then
      	echo "in if"
      fi
      
    • 语句块

        在某些只允许使用单个语句的地方,要想使用多条语句,可以放到花括号中构建一个语句块。

      get_confirm && { # and列表中使用语句块
          echo "hello"
          cat test.txt
      }
      

    (4)函数

      要在shell脚本中使用函数,只需要写出函数名,然后跟一对括号,再把函数中的语句放在一对花括号中,并且把函数定义放到函数调用之前。函数体内可以用local关键字声明局部变量。

      函数参数:当一个函数被调用时,脚本程序的位置参数($*,$@,$#,$1,$2等等)会被替换为函数的参数,当函数执行完毕后,这些参数会恢复为先前的值。下面的例子展示了函数的用法。

    #!/bin/sh
    yes_or_no(){    # 函数定义
        echo "Is your name $* ?"
        while true
        do
        	echo -n "Enter yes or no: "
        	read x
        	case "$x" in
        		y | yes ) return 0;;
        		n | no ) return 1;;
        		* ) echo "Answer yes or no"
        	esac
        done
    }
    # 以下是主程序部分:
    echo  "Original parameters are $*"
    if yes_or_no "$1"
    then
    	echo "Hi $1,nice name"
    else
    	echo "Never mind"
    fi
    exit 0
    

    (5)命令

    • break命令和continue命令

        这两条命令比较简单,break应用于跳出for、while、until循环,continue命令用于跳过当前这一次循环。

    • : 命令

        :相当于一个空命令,或者相当于true的别名,例如 while : 就代表一个死循环

    • . 命令

        . 命令用于在当前shell中执行命令,如前面用到的./first.sh

    • echo 命令和read命令

        echo命令用于输出结尾带有换行符的字符串,前面已多次用到,它还有两个常用的参数,如下所示。现在最新版本的shell常常用printf来代替echo。read命令用于将用户的输入赋给一个变量。

      $ echo -n "string to output" # 去掉换行符
      $ echo -e "string to outputc" # -e确保启用了反斜杠转移字符
      
    • eval 命令

        eval命令用于对参数进行求值,它是shell的内置命令,通常不会以单独命令的形式存在,以下的例子展示eval的用法。eval命令就像一个额外的$,它给出一个变量的值的值。

      foo=10
      x=foo
      y='$'$x # $x就是foo
      echo $y # 输出的是$foo
      eval z='$'$x
      echo $z # 输出的是10
      
    • exec 命令

        exec命令时执行一个shell程序,也就是将当前shell替换为一个不同的程序,当前脚本程序exec命令之后的代码都不会执行。还有一种用法是,它可以用于修改文件描述符,比较少见。

    • exit n 命令

        exit命令使脚本程序以退出码n结束运行。退出码0表示成功,1~125是错误代码,126代表文件不可执行,127代表命令未找到。

    • export 命令

        export命令将自己的参数建为一个环境变量,而这个环境变量可以被当前程序调用的其他脚本或者程序看到。

      # 以下是export2.sh
      #!/bin/sh
      echo $foo
      echo $bar
      
      # 以下是export1.sh
      #!/bin/sh
      foo = "foo foo foo"
      export bar = "bar bar bar"
      export2 # 调用脚本2
      # 此时如果执行脚本1,我们会看到输出bar的值,因为被声明为环境变量,在脚本2中可见,而foo不会被输出
      
    • expr 命令和$( )$(( ))

        expr命令将它的参数作为表达式来求值,最常见用法就是进行数学运算。$(command)的结果就是执行command的 输出结果,和两个反引号的功能相同。

        对于表达式求值,一种更新的方法是使用$(( ))命令,将求值的表达式放到$(( ))中,可以很简单的完成数学运算,常见的表达式求值有:加+、减-、乘*、除/、取模%、与&、或|、等于=、不等!=、大于>、小于<等等。

        以下的例子展示这几个命令的用法:

      $ x=1
      $ x=`expr $x + 1` # 反引号是x取值为xpr $x + 1的结果
      $ x=$(expr $x + 1)  # $(command)具有和反引号相同的功能
      $ x=$(($x + 1)) # 一种更新的方法是使用$(( ))命令,代替expr命令
      
    • set 命令和unset 命令

        set命令的作用是为shell设置参数变量。unset命令的作用是从环境中删除变量或者函数。

      #!/bin/sh
      echo the date is $(date)
      set $(date) # 将date设置为参数变量
      echo the month is $2 # 输出第二个位置参数,月份
      foo = 0
      unset foo # 删除变量foo
      exit 0
      
    • shift 命令

        shift命令把所有的参数变量左移一个位置,使得$2变为$1$3变为$2,依次类推,原来的$1被丢弃,而$0仍将保持不变。如果指定数值参数,可以左移相应的次数。该命令的一个主要作用是用来扫描参数,如下所示。

      #!/bin/sh
      while [ "$1" != "" ]; do
      	echo "$1"
      	shift
      done
      exit 0
      
    • trap 命令

        trap命令用于指定在接收到相应的信号后将要采取的行动,trap命令的常见形式如下:

      trap command signal
      # 第一个参数是接收到信号时将要采取的行动
      # 第二个参数是要处理的信号名
      
    • find 命令

        find命令是一个很有用的命令,主要用于在系统中找文件,也就是文件搜索。其基本语法格式如下:

      find [path] [options] [tests] [actions]
      

        其中,path部分很好理解,是要搜索的路径,可以是相对路径,也可以是绝对路径,也可以是多个路径。

        options部分是一些可用选项,主要有-depth(在查看目录本身之前先搜索目录的内容),-follow(跟随符号链接),-maxdepths N (最多搜索N层目录),-mount(或者-xdev,指不搜索其他文件系统中的目录)

        tests是测试部分,每种测试的返回结果是true或false,主要有以下几种:-atime N (文件在N天前被最后访问过),-mtime N (文件在N天前被最后修改过),-name(文件名匹配),-newer otherfile(文件比otherfile要新),-type c(文件类型是c、d、f,分别对应特殊字符文件、目录、普通文件),-user username(文件的拥有者是username)。

        注意:tests测试可以组合使用,有三个组合命令:-and,-or,-not

        actions部分是匹配之后要执行的动作,比如:-exec command(执行一条命令,最常见),-print(打印)等等。

      $ find / -name test -print
      $ find / -mount -name test -print
      $ find . -newer while2 -print
      $ find . -newer while2 -type f -print
      
    • grep 命令

        grep命令是通用正则表达式解析器,通俗的说,find命令在系统中找文件,grep命令在文件中找字符串,一种常用的做法是将grep作为传递给-exec的一条命令。

        grep命令的基本语法如下:

      grep [options] PATTERN [FILES]
      

        options是一些主要选项,常用的有-c (输出匹配行的数目)、-E(启用扩展表达式)、-h(取消每个输出行的普通前缀)、-i(忽略大小写)、-l(只列出包含匹配行的文件名)、-v(取反,搜索不匹配行)。

        PATTERN主要是一些匹配模式,常用正则表达式来表示,关于正则表达式的内容参照另一篇博文正则表达式

      grep in words.txt
      grep -c in words.txt words2.txt
      grep -c -v in words.txt words2.txt
      grep "e$" words2.txt
      grep "a[[:blank:]]" words2.txt
      grep -E [a-z]{10} words2.txt
      

    总结

      Shell程序设计作为一种脚本语言,在Linux系统中有广泛的应用,本文记录了关于Shell程序设计的基础语法知识和常用命令,方便查询,熟练使用shell也需要经常实践,这对于完成一些较简单的编程任务很有帮助。另外,shell程序在Linux中海油一个可视化工具:dialog,由于不常使用,这里不再介绍。

  • 相关阅读:
    firefly rk3399 增加 HL340 驱动(编译内核)
    STM32移植ROS发布超声波信息
    路径规划基础A*算法
    ROS融合IMU笔记
    a2 任意角度选取设置
    如何用代码设置机器人初始坐标实现 2D Pose Estimate功能
    APP 链接ROS时出现pymongo.errors.ServerSelectionTimeoutError: localhost:27017 错误
    基于opencv+python的二维码识别
    SAP UI5学习笔记之(二)熟悉的HelloWorld
    SAP UI5学习笔记之(一)初识SAP UI5
  • 原文地址:https://www.cnblogs.com/gzshan/p/10783806.html
Copyright © 2020-2023  润新知