zoukankan      html  css  js  c++  java
  • linux shell编程

    可以通过使用shell使大量的任务自动化,shell特别擅长系统管理任务,尤其适合那些易用性、可维护性和便携性比效率更重要的任务。

      下面,让我们一起来看看shell是如何工作的:

      建立一个脚本

      Linux中有好多中不同的shell,但是通常我们使用bash (bourne again shell) 进行shell编程,因为bash是免费的并且很容易使用。所以在本文中笔者所提供的脚本都是使用bash(但是在大多数情况下,这些脚本同样可以在 bash的大姐,bourne shell中运行)。

      如同其他语言一样,通过我们使用任意一种文字编辑器,比如nedit、kedit、emacs、vi等来编写我们的shell程序。程序必须以下面的行开始(必须方在文件的第一行):

    #!/bin/sh

      符号#!用来告诉系统它后面的参数是用来执行该文件的程序。在这个例子中我们使用/bin/sh来执行程序。当编辑好脚本时,如果要执行该脚本,还必须使其可执行。

      要使脚本可执行:

    chmod +x filename

      然后,您可以通过输入: ./filename 来执行您的脚本。

      注释

      在进行shell编程时,以#开头的句子表示注释,直到这一行的结束。我们真诚地建议您在程序中使用注释。如果您使用了注释,那么即使相当长的时间内没有使用该脚本,您也能在很短的时间内明白该脚本的作用及工作原理。

      变量

      在其他编程语言中您必须使用变量。在shell编程中,所有的变量都由字符串组成,并且您不需要对变量进行声明。要赋值给一个变量,您可以这样写:

    变量名=值

      取出变量值可以加一个美元符号($)在变量前面:

    #!/bin/sh

      #对变量赋值:

    a="hello world"

      # 现在打印变量a的内容:

    echo "A is:"
    echo $a

      在您的编辑器中输入以上内容,然后将其保存为一个文件first。之后执行chmod +x first。使其可执行,最后输入./first执行该脚本。

      这个脚本将会输出:

    A is:
      hello world

      有时候变量名很容易与其他文字混淆,比如:

    num=2
      echo "this is the $numnd"

      这并不会打印出"this is the 2nd",而仅仅打印"this is the ",因为shell会去搜索变量numnd的值,但是这个变量时没有值的。可以使用花括号来告诉shell我们要打印的是num变量:

    num=2
      echo "this is the $nd"

      这将打印: this is the 2nd

      有许多变量是系统自动设定的,这将在后面使用这些变量时进行讨论。

     如果您需要处理数学表达式,那么您需要使用诸如expr等程序(见下面)。除了一般的仅在程序内有效的shell变量以外,还有环境变量。由export关键字处理过的变量叫做环境变量。我们不对环境变量进行讨论,因为通常情况下仅仅在登录脚本中使用环境变量。

      Shell命令和流程控制

      在shell脚本中可以使用三类命令:

      1)Unix 命令:

      虽然在shell脚本中可以使用任意的unix命令,但是还是由一些相对更常用的命令。这些命令通常是用来进行文件和文字操作的。

     常用命令语法及功能:

      echo "some text": 将文字内容打印在屏幕上。

    ls: 文件列表。

      wc –l file wc -w file wc -c file: 计算文件行数 计算文件中的单词数 计算文件中的字符数。

      cp sourcefile destfile: 文件拷贝。

      mv oldname newname : 重命名文件或移动文件。

      rm file: 删除文件。

      grep 'pattern' file: 在文件内搜索字符串比如:grep 'searchstring' file.txt

      cut -b colnum file: 指定欲显示的文件内容范围,并将它们输出到标准输出设备比如:输出每行第5个到第9个字符cut –b 5-9 file.txt千万不要和cat命令混淆,这是两个完全不同的命令。

      cat file.txt: 输出文件内容到标准输出设备(屏幕)上。

      file somefile: 得到文件类型。

      read var: 提示用户输入,并将输入赋值给变量。

      sort file.txt: 对file.txt文件中的行进行排序。

      uniq: 删除文本文件中出现的行列比如: sort file.txt | uniq。

      expr: 进行数学运算Example: add 2 and 3 expr 2 "+" 3。

      find: 搜索文件比如:根据文件名搜索find . -name filename -print。

      tee: 将数据输出到标准输出设备(屏幕) 和文件比如:somecommand | tee outfile。

      basename file: 返回不包含路径的文件名比如: basename /bin/tux将返回 tux。

      dirname file: 返回文件所在路径比如:dirname /bin/tux将返回 /bin。

      head file: 打印文本文件开头几行。

      tail file : 打印文本文件末尾几行。

      sed: Sed是一个基本的查找替换程序。可以从标准输入(比如命令管道)读入文本,并将结果输出到标准输出(屏幕)。该命令采用正则表达式(见参考)进行搜索。不要和shell中的通配符相混淆。比如:将linuxfocus 替换为 LinuxFocus :cat text.file | sed 's/linuxfocus/LinuxFocus/' > newtext.file。

    awk: awk 用来从文本文件中提取字段。缺省地,字段分割符是空格,可以使用-F指定其他分割符。cat file.txt | awk -F, '{print "," }'这里我们使用,作为字段分割符,同时打印第一个和第三个字段。如果该文件内容如下:Adam Bor, 34, IndiaKerry Miller, 22, USA

      命令输出结果为:

    Adam Bor, IndiaKerry Miller.

      2) 概念: 管道, 重定向和 backtick

      这些不是系统命令,但是他们真的很重要。

      管道 (|) 将一个命令的输出作为另外一个命令的输入。

    grep "hello" file.txt | wc -l

      在file.txt中搜索包含有”hello”的行并计算其行数。在这里grep命令的输出作为wc命令的输入。当然您可以使用多个命令。

      重定向:将命令的结果输出到文件,而不是标准输出(屏幕)。

      > 写入文件并覆盖旧文件。

      >> 加到文件的尾部,保留旧文件内容。

      反短斜线,使用反短斜线可以将一个命令的输出作为另外一个命令的一个命令行参数。

      命令:

    find . -mtime -1 -type f -print

      用来查找过去24小时(-mtime –2则表示过去48小时)内修改过的文件。如果您想将所有查找到的文件打一个包,则可以使用以下脚本:

    #!/bin/sh
      # The ticks are backticks (`) not normal quotes ('):
      tar -zcvf lastmod.tar.gz `find . -mtime -1 -type f -print`

      3) 流程控制

      "if" 表达式 如果条件为真则执行then后面的部分:

      if ....; then

    ________________________________________________________________________________________________________

    Shell基本语法

      像高级程序设计语言一样,Shell也提供说明和使用变量的功能。对Shell来讲,所有变量的取值都是一个字串,Shell程序采用$var的形式来引用名为var的变量的值。

      Shell有以下几种基本类型的变量。

      (1)Shell定义的环境变量:

      Shell在开始执行时就已经定义了一些和系统的工作环境有关的变量,用户还可以重新定义这些变量,常用的Shell环境变量有:

      HOME 用于保存注册目录的完全路径名。

      PATH 用于保存用冒号分隔的目录路径名,Shell将按PATH变量中给出的顺序搜索这些目录,找到的第一个与命令名称一致的可执行文件将被执行。

      TERM 终端的类型。

      UID 当前用户的识别字,取值是由数位构成的字串。

      PWD 当前工作目录的绝对路径名,该变量的取值随cd命令的使用而变化。

      PS1 主提示符,在特权用户下,默认的主提示符是#,在普通用户下,默认的主提示符是$。

      PS2 在Shell接收用户输入命令的过程中,如果用户在输入行的末尾输入“\”然后回车,或者当用户按回车键时Shell判断出用户输入的命令没有结束时,就显示这个辅助提示符,提示用户继续输入命令的其余部分,默认的辅助提示符是>。

      (2)用户定义的变量:

      用户可以按照下面的语法规则定义自己的变量:

      变量名=变量值

      要注意的一点是,在定义变量时,变量名前不应加符号$,在引用变量的内容时则应在变量名前加$;在给变量赋值时,等号两边一定不能留空格,若变量中本身就包含了空格,则整个字串都要用双引号括起来。

      在编写Shell程序时,为了使变量名和命令名相区别,建议所有的变量名都用大写字母来表示。

      有时我们想要在说明一个变量并对它设置为一个特定值后就不在改变它的值时,可以用下面的命令来保证一个变量的只读性:

      readonly 变量名

      在任何时候,创建的变量都只是当前Shell的局部变量,所以不能被Shell运行的其他命令或Shell程序所利用,而export命令可以将一个局部变量提供给Shell执行的其他命令使用,其格式为:

      export 变量名

      也可以在给变量赋值的同时使用export命令:

      export 变量名=变量值

      使用export说明的变量,在Shell以后运行的所有命令或程序中都可以访问到。

      (3)位置参数:

      位置参数是一种在调用Shell程序的命令行中按照各自的位置决定的变量,是在程序名之后输入的参数。位置参数之间用空格分隔,Shell取第一个位置参数替换程序文件中的$1,第二个替换$2,依次类推。$0是一个特殊的变量,它的内容是当前这个Shell程序的文件名,所以,$0不是一个位置参数,在显示当前所有的位置参数时是不包括$0的。

      (4)预定义变量:

      预定义变量和环境变量相类似,也是在Shell一开始时就定义了的变量。所不同的是,用户只能根据Shell的定义来使用这些变量,而不能重定义它。所有预定义变量都是由$符和另一个符号组成的,常用的Shell预定义变量有:

      $# 位置参数的数量。

      $* 所有位置参数的内容。

      $? 命令执行后返回的状态。

      $$ 当前进程的进程号。

      $! 后台运行的最后一个进程号。

      $0 当前执行的进程名。

      其中,$?用于检查上一个命令执行是否正确。(在Linux中,命令退出状态为0表示该命令正确执行,任何非0值表示命令出错。)

      $$变量最常见的用途是用做暂存文件的名字以保证暂存文件不会重复。

      (5)参数置换的变量:

      Shell提供了参数置换功能以便用户可以根据不同的条件来给变量赋不同的值。参数置换的变量有4种,这些变量通常与某一个位置参数相联系,根据指定的位置参数是否已经设置类决定变量的取值,它们的语法和功能分别如下。

      a. 变量=${参数-word}:如果设置了参数,则用参数的值置换变量的值,否则用word置换。即这种变量的值等于某一个参数的值,如果该参数没有设置,则变量就等于word的值。

      b. 变量=${参数=word}:如果设置了参数,则用参数的值置换变量的值,否则把变量设置成word,然后再用word替换参数的值。注意,位置参数不能用于这种方式,因为在Shell程序中不能为位置参数赋值。

      c. 变量=${参数?word}:如果设置了参数,则用参数的值置换变量的值,否则就显示word并从Shell中退出,如果省略了word,则显示标准信息。这种变量要求一定等于某一个参数的值。如果该参数没有设置,就显示一个信息,然后退出,因此这种方式常用于出错指示。

      d. 变量=${参数+word}:如果设置了参数,则用word置换变量,否则不进行置换。

      所有这4种形式中的“参数”既可以是位置参数,也可以是另一个变量,只是用位置参数的情况比较多。

    Shell程序设计的流程控制

      和其他高级程序设计语言一样,Shell提供了用来控制程序执行流程的命令,包括条件分支和循环结构,用户可以用这些命令创建非常复杂的程序。

      与传统语言不同的是,Shell用于指定条件值的不是布尔运算式,而是命令和字串。

      1.测试命令

      test命令用于检查某个条件是否成立,它可以进行数值、字符和文件3个方面的测试,其测试符和相应的功能分别如下。

      (1)数值测试:

      -eq 等于则为真。

      -ne 不等于则为真。

      -gt 大于则为真。

      -ge 大于等于则为真。

      -lt 小于则为真。

      -le 小于等于则为真。

      (2)字串测试:

      = 等于则为真。

      != 不相等则为真。

      -z字串 字串长度伪则为真。

      -n字串 字串长度不伪则为真。

      (3)文件测试:

      -e文件名 如果文件存在则为真。

      -r文件名 如果文件存在且可读则为真。

      -w文件名 如果文件存在且可写则为真。

      -x文件名 如果文件存在且可执行则为真。

      -s文件名 如果文件存在且至少有一个字符则为真。

      -d文件名 如果文件存在且为目录则为真。

      -f文件名 如果文件存在且为普通文件则为真。

      -c文件名 如果文件存在且为字符型特殊文件则为真。

      -b文件名 如果文件存在且为块特殊文件则为真。

      另外,Linux还提供了与(!)、或(-o)、非(-a)三个逻辑操作符,用于将测试条件连接起来,其优先顺序为:!最高,-a次之,-o最低。

      同时,bash也能完成简单的算术运算,格式如下:

      $[expression]

      例如:

      var1=2

      var2=$[var1*10+1]

      则var2的值为21。

      2.if条件语句

      Shell程序中的条件分支是通过if条件语句来实现的,其一般格式为:

      if 条件命令串

      then

      条件为真时的命令串

      else

      条件为假时的命令串

      fi

      3.for循环

      for循环对一个变量的可能的值都执行一个命令序列。赋给变量的几个数值既可以在程序内以数值列表的形式提供,也可以在程序以外以位置参数的形式提供。for循环的一般格式为:

      for变量名    [in数值列表]

      do

      若干个命令行

      done

      变量名可以是用户选择的任何字串,如果变量名是var,则在in之后给出的数值将顺序替换循环命令列表中的$var。如果省略了in,则变量var的取值将是位置参数。对变量的每一个可能的赋值都将执行do和done之间的命令列表。

      4.while和until循环

      while和until命令都是用命令的返回状态值来控制循环的。While循环的一般格式为:

      while

      若干个命令行1

      do

      若干个命令行2

      done

      只要while的“若干个命令行1”中最后一个命令的返回状态为真,while循环就继续执行do...done之间的“若干个命令行2”。

      until命令是另一种循环结构,它和while命令相似,其格式如下:

      until

       若干个命令行1

      do

       若干个命令行2

      done

      until循环和while循环的区别在于:while循环在条件为真时继续执行循环,而until则是在条件为假时继续执行循环。

      Shell还提供了true和false两条命令用于创建无限循环结构,它们的返回状态分别是总为0或总为非0。

      5.case条件选择

      if条件语句用于在两个选项中选定一项,而case条件选择为用户提供了根据字串或变量的值从多个选项中选择一项的方法,其格式如下:

      case string in

      exp-1)

      若干个命令行1

      ;;

      exp-2)

       若干个命令行2

      ;;

      ……

      *)

      其他命令行

      esac

      Shell通过计算字串string的值,将其结果依次和运算式exp-1, exp-2等进行比较,直到找到一个匹配的运算式为止。如果找到了匹配项,则执行它下面的命令直到遇到一对分号(;;)为止。

      在case运算式中也可以使用Shell的通配符(“*”、“?”、“[ ]”)。通常用 * 作为case命令的最后运算式以便在前面找不到任何相应的匹配项时执行“其他命令行”的命令。

      6.无条件控制语句break和continue

      break用于立即终止当前循环的执行,而contiune用于不执行循环中后面的语句而立即开始下一个循环的执行。这两个语句只有放在do和done之间才有效。

      7.函数定义

      在Shell中还可以定义函数。函数实际上也是由若干条Shell命令组成的,因此它与Shell程序形式上是相似的,不同的是它不是一个单独的进程,而是Shell程序的一部分。函数定义的基本格式为:

      functionname

      {

       若干命令行

      }

      调用函数的格式为:

      functionname param1 param2…

      Shell函数可以完成某些例行的工作,而且还可以有自己的退出状态,因此函数也可以作为if, while等控制结构的条件。

      在函数定义时不用带参数说明,但在调用函数时可以带有参数,此时Shell将把这些参数分别赋予相应的位置参数$1, $2, ...及$*。

      8.命令分组

      在Shell中有两种命令分组的方法:()和{}。前者当Shell执行()中的命令时将再创建一个新的子进程,然后这个子进程去执行圆括弧中的命令。当用户在执行某个命令时不想让命令运行时对状态集合(如位置参数、环境变量、当前工作目录等)的改变影响到下面语句的执行时,就应该把这些命令放在圆括弧中,这样就能保证所有的改变只对子进程产生影响,而父进程不受任何干扰。{}用于将顺序执行的命令的输出结果用于另一个命令的输入(管道方式)。当我们要真正使用圆括弧和花括弧时(如计算运算式的优先顺序),则需要在其前面加上转义符(\)以便让Shell知道它们不是用于命令执行的控制所用。

      9.信号

      trap命令用于在Shell程序中捕捉信号,之后可以有3种反应方式:

      (1)执行一段程序来处理这一信号。

      (2)接受信号的默认操作。

      (3)忽视这一信号。

      trap对上面3种方式提供了3种基本形式:

      第一种形式的trap命令在Shell接收到与signal list清单中数值相同的信号时,将执行双引号中的命令串。

      trap 'commands' signal-list

      trap "commands" signal-list

      为了恢复信号的默认操作,使用第二种形式的trap命令:

      trap signal-list

      第三种形式的trap命令允许忽略信号:

      trap " " signal-list

      注意:

      (1)对信号11(段违例)不能捕捉,因为Shell本身需要捕捉该信号去进行内存的转储。

      (2)在trap中可以定义对信号0的处理(实际上没有这个信号),Shell程序在其终止(如执行exit语句)时发出该信号。

      (3)在捕捉到signal-list中指定的信号并执行完相应的命令之后,如果这些命令没有将Shell程序终止的话,Shell程序将继续执行收到信号时所执行的命令后面的命令,这样将很容易导致Shell程序无法终止。

      另外,在trap语句中,单引号和双引号是不同的。当Shell程序第一次碰到trap语句时,将把commands中的命令扫描一遍。此时若 commands是用单引号括起来的话,那么Shell不会对commands中的变量和命令进行替换,否则commands中的变量和命令将用当时具体的值来替换。

    运行Shell程序的方法

      用户可以用任何编辑程序来编写Shell程序。因为Shell程序是解释执行的,所以不需要编译成目的程序。按照Shell编程的惯例,以 bash为例,程序的第一行一般为“#!/bin/bash”,其中 # 表示该行是注释,叹号 !告诉Shell运行叹号之后的命令并用文档的其余部分作为输入,也就是运行/bin/bash并让/bin/bash去执行Shell程序的内容。

      执行Shell程序的方法有3种。

      1.sh Shell程序文件名

      这种方法的命令格式为:

      bash Shell程序文件名

      这实际上是调用一个新的bash命令解释程序,而把Shell程序文件名作为参数传递给它。新启动的Shell将去读指定的文件,可执行文件中列出的命令,当所有的命令都执行完后结束。该方法的优点是可以利用Shell调试功能。

      2.sh

      格式为:

      bash< Shell程序名

      这种方式就是利用输入重定向,使Shell命令解释程序的输入取自指定的程序文件。

      3.用chmod命令使Shell程序成为可执行的

      一个文件能否运行取决于该文档的内容本身可执行且该文件具有执行权。对于Shell程序,当用编辑器生成一个文件时,系统赋予的许可权都是644(rw-r-r--),因此,当用户需要运行这个文件时,只需要直接键入文件名即可。

      在这3种运行Shell程序的方法中,最好按下面的方式选择:当刚创建一个Shell程序,对它的正确性还没有把握时,应当使用第一种方式进行调试。当一个Shell程序已经调试好时,应使用第三种方式把它固定下来,以后只要键入相应的文件名即可,并可被另一个程序所调用。

      4.bash程序的调试

      在编程过程中难免会出错,有的时候,调试程序比编写程序花费的时间还要多,Shell程序同样如此。

      Shell程序的调试主要是利用bash命令解释程序的选择项。调用bash的形式是:

      bash -选择项Shell程序文件名

      几个常用的选择项是:

      -e 如果一个命令失败就立即退出。

      -n 读入命令但是不执行它们。

      -u 置换时把未设置的变量看做出错。

      -v 当读入Shell输入行时把它们显示出来。

      -x 执行命令时把命令和它们的参数显示出来。

      上面的所有选项也可以在Shell程序内部用“set -选择项”的形式引用,而“set +选择项”则将禁止该选择项起作用。如果只想对程序的某一部分使用某些选择项时,则可以将该部分用上面两个语句包围起来。

      (1)未置变量退出和立即退出

      未置变量退出特性允许用户对所有变量进行检查,如果引用了一个未赋值的变量就终止Shell程序的执行。Shell通常允许未置变量的使用,在这种情况下,变量的值为空。如果设置了未置变量退出选择项,则一旦使用了未置变量就显示错误信息,并终止程序的运行。未置变量退出选择项为-u。

      当Shell运行时,若遇到不存在或不可执行的命令、重定向失败或命令非正常结束等情况时,如果未经重新定向,该出错信息会显示在终端屏幕上,而Shell程序仍将继续执行。要想在错误发生时迫使Shell程序立即结束,可以使用-e选项将Shell程序的执行立即终止。

      (2)Shell程序的跟踪

      调试Shell程序的主要方法是利用Shell命令解释程序的-v或-x选项来跟踪程序的执行。-v选择项使Shell在执行程序的过程中,把它读入的每一个命令行都显示出来,而-x选择项使Shell在执行程序的过程中把它执行的每一个命令在行首用一个+加上命令名显示出来。并把每一个变量和该变量所取的值也显示出来。因此,它们的主要区别在于:在执行命令行之前无-v,则显示出命令行的原始内容,而有-v时则显示出经过替换后的命令行的内容。

      除了使用Shell的-v和-x选择项以外,还可以在Shell程序内部采取一些辅助调试的措施。例如,可以在Shell程序的一些关键地方使用echo命令把必要的信息显示出来,它的作用相当于C语言中的printf语句,这样就可以知道程序运行到什么地方及程序目前的状态。

    bash的内部命令

      bash命令解释套装程序包含了一些内部命令。内部命令在目录列表时是看不见的,它们由Shell本身提供。常用的内部命令有:echo, eval, exec, export, readonly, read, shift, wait和点(.)。下面简单介绍其命令格式和功能。

      1.echo

      命令格式:echo arg

      功能:在屏幕上显示出由arg指定的字串。

      2.eval

      命令格式:eval args

      功能:当Shell程序执行到eval语句时,Shell读入参数args,并将它们组合成一个新的命令,然后执行。

      3.exec

      命令格式:exec命令参数

      功能:当Shell执行到exec语句时,不会去创建新的子进程,而是转去执行指定的命令,当指定的命令执行完时,该进程(也就是最初的Shell)就终止了,所以Shell程序中exec后面的语句将不再被执行。

      4.export

      命令格式:export变量名 或:export变量名=变量值

      功能:Shell可以用export把它的变量向下带入子Shell,从而让子进程继承父进程中的环境变量。但子Shell不能用export把它的变量向上带入父Shell。

      注意:不带任何变量名的export语句将显示出当前所有的export变量。

      5.readonly

      命令格式:readonly变量名

      功能:将一个用户定义的Shell变量标识为不可变。不带任何参数的readonly命令将显示出所有只读的Shell变量。

      6.read

      命令格式:read变量名表

      功能:从标准输入设备读入一行,分解成若干字,赋值给Shell程序内部定义的变量。

      7.shift语句

      功能:shift语句按如下方式重新命名所有的位置参数变量,即$2成为$1,$3成为$2…在程序中每使用一次shift语句,都使所有的位置参数依次向左移动一个位置,并使位置参数$#减1,直到减到0为止。

      8.wait

      功能:使Shell等待在后台启动的所有子进程结束。wait的返回值总是真。

      9.exit

      功能:退出Shell程序。在exit之后可有选择地指定一个数位作为返回状态。

      10.“.”(点)

      命令格式:. Shell程序文件名

      功能:使Shell读入指定的Shell程序文件并依次执行文件中的所有语句。

  • 相关阅读:
    简单的PHP+SMARTY分页类
    三大WEB服务器对比分析(apache ,lighttpd,nginx)
    逐步设置vim C/C++语法高亮显示和自动缩进
    Lighttpd 的安装配置(web服务器软件)
    ASP如何获取文件和目录空间大小
    innerHtml用法
    临时表操作的一些见解(解决了我在存储过程中使用临时表的困惑)
    Javascript的IE和Firefox兼容性汇编
    Javascript的IE和Firefox兼容性汇编
    innerHtml用法
  • 原文地址:https://www.cnblogs.com/buffer/p/1607637.html
Copyright © 2011-2022 走看看