Linux经常使用命令

  • 登录时显示信息放在/etc/motd和/etc/profile.d/xxx.sh

    • motd放置字符串
    • profile.d下放置脚本文件
  • echo

    • echo -e 处理特殊字符。若字符串中出现下面字符。则特别加以处理,而不会将它当成一般文字输出:

      • \a 发出警告声;
      • \b 删除前一个字符。
      • \c 最后不加上换行符号;
      • \f 换行但光标仍旧停留在原来的位置。
      • \n 换行且光标移至行首。
      • \r 光标移至行首。但不换行;
      • \t 插入tab;
      • \v 与\f同样;
      • \ 插入\字符;
      • \nnn 插入nnn(八进制)所代表的ASCII字符;
    • echo -n 不换行输出
  • date +%T 以格式化输出当前时间

  • ls -lrt 按时间升序列表显示
  • dirname 路径 dirname $0
  • basename 文件名称
  • 在环境变量文件里使用alias vi=’vim’就可以实现vi取代vim命令

Shell

Shell脚本的建立与运行

Shell 脚本的建立

  • 使用文本编辑器编辑脚本文件

    • vi script-file
  • 为脚本文件加入可运行权限
    • chmod +x script-file

Shell 脚本的运行

  • 在子Shell中运行

    • 变量及结果在结束时从内存销毁,外部shell无法引用
    • bash script-file
    • 或者./script-file
  • 在当前Shell中运行
    • 当前shell可继续引用运行结束shell的变量及结果
    • source script-file
    • . script-file

Shell脚本的编码规范

以 #! 开头:通知系统用何解释器运行此脚本

  • #!/bin/bash
  • 或者#!/bin/sh

以凝视形式说明例如以下的内容:

  • # 脚本名称
  • # 脚本功能
  • # 作者及联系方式
  • # 版本号更新记录
  • # 版权声明
  • # 对算法做简要说明(假设是复杂脚本)
    • 演示样例

      1. #!/bin/bash
      2. # This is the first Bash shell program
      3. # Scriptname: greetings.sh
      4. echo
      5. echo -e "Hello $LOGNAME, \c"
      6. echo "it's nice talking to you."
      7. echo -n "Your present working directory is: "
      8. pwd # Show the name of present directory
      9. echo
      10. echo -e "The time is `date +%T`!. \nBye"
      11. echo
    • 时间同步shell脚本

      1. #!/bin/bash
      2. ## Script Name:/etc/cron.daily/ntpdate
      3. # 使用NTP的client命令ntpdate与远程NTPserver进行同步
      4. # 也能够用局域网内的NTPserver替换 pool.ntp.org
      5. /usr/sbin/ntpdate -s pool.ntp.org
      6. # 更改硬件时钟时都会记录在/etc/adjtime文件里
      7. # 使hwclock依据先前的记录来估算硬件时钟的偏差。
      8. # 并用来校正眼下的硬件时钟
      9. /sbin/hwclock --adjust
      10. # 将系统时钟同步到硬件时钟
      11. /sbin/hwclock systohc

脚本调试方法

在 bash 调用脚本时使用參数

  • bash [-x] [-n] [-v] scriptName

    • sh –x 脚本名

      • 该选项能够使用户跟踪脚本的运行。此时 shell 对脚本中每条命令的处理过程为:先运行替换,然后显示,再运行它。shell 显示脚本中的行时。会在行首加入一个加号 “ + ”
      • 以调试模式运行脚本
    • sh –v 脚本名
      • 在运行脚本之前,按输入的原样打印脚本中的各行
      • 显示脚本中每一个原始命令行及其运行结果
    • sh –n 脚本名
      • 对脚本进行语法检查。但不运行脚本。

        假设存在语法错误,shell 会报错。假设没有错误,则不显示不论什么内容

      • 对脚本进行语法检查

在脚本中使用 bash 内置的 set 命令使整个或部分脚本处于调试模式

  • 开启:set [-x] [-n] [-v]

    • 在脚本内使用set命令开启调试选项

      • set -x :显示由shell运行的命令及其參数
      • set -v :显示由shell读入的命令行
      • set -n :读取命令但不运行他们。用于语法检查
  • 结束:set [+x] [+n] [+v]
    • 在脚本内使用set命令关闭已开启的调试选项

      • set +x
      • set +v
      • set +n
  • 演示样例

    1. #!/bin/bash
    2. # This is the first Bash shell program
    3. # Scriptname: greetings.sh
    4. set -x ### Turn ON debug mode ###
    5. echo
    6. echo -e "Hello $LOGNAME, \c"
    7. echo "it's nice talking to you."
    8. echo -n "Your present working directory is: "
    9. pwd # Show the name of present directory
    10. echo
    11. set +x ### Turn OFF debug mode ###
    12. echo -e "The time is `date +%T`!. \nBye"
    13. echo

Shell脚本的类型

非交互式脚本

  • 不须要读取用户的输入, 也不用向用户反馈某些信息
  • 每次运行都是可预见的, 由于它不读取用户输入, 參数是固定的
  • 能够在后台运行

交互式脚本

  • 脚本能够读取用户的输入, 实时向用户反馈信息(输出某些信息)
  • 这种脚本更灵活, 每次运行时的參数可由用户动态设定
  • 用户界面更友好,但不适用于自己主动化任务(如cron任务)

Shell变量操作

  • 变量替换扩展

    • 变量測试
    • 变量的字符串操作
    • 变量的间接引用
  • 变量的数值计算
    • $[expression]
    • $((expression))
    • expr
    • let
    • declare -i
  • 输入
    • 变量赋值

      • name=value
      • readonly
    • 从标准输入读取
      • read
  • 输出
    • echo
    • printf

变量替换扩展

变量測试

演示样例

字符串计数、截取

演示样例

字符串替换

演示样例

变量的间接引用

  • 通过 str2 的值来引用 str1 的值

    • 错误演示样例

      1. str1="Hello World"
      2. str2=str1
      3. echo $str2
    • 正确演示样例

      1. str1="Hello World"
      2. str2=str1
      3. newstr=${!str2}
      4. echo $newstr
      5. Hello World

      6. echo ${!str2}
      7. Hello World

      8. eval newstr=\$$str2
      9. echo $newstr
      10. Hello World

      11. eval echo \$$str2
      12. Hello World
  • x=”CENTOS”

  • CENTOS_URL=”http://mirrors.163.com/centos/”
  • 通过 x 的值来引用 CENTOS_URL 的值

    1. bash2.0以上才支持
    2. newstr=${x}_URL
    3. echo $newstr
    4. CENTOS_URL
    5. echo ${!newstr}
    6. http://mirrors.163.com/centos/

    7. eval newstr=\$${x}_URL
    8. echo $newstr

    9. eval echo \$${x}_URL

Shell内置命令——eval

eval arg1 [arg2] … [argN]

* 对參数进行两次扫描和替换

* 将全部的參数连接成一个表达式。并计算或运行该表达式

* 參数中的不论什么变量都将被展开

  1. listpage="ls -l | more"
  2. eval $listpage
  3. eval $(ssh-agent)
  4. eval newstr=\$$str2
  5. eval echo \$${x}_URL

Shell 变量的分类

  • 用户自定义变量

    • 由用户自定义、改动和使用
  • Shell 环境变量
    • 由系统维护,用于设置用户的Shell工作环境
    • 仅仅有少数的变量用户能够改动其值
  • 位置參数变量(Positional Parameters)
    • 通过命令行给程序传递运行參数
    • 可用 shift 命令实现位置參数的迁移
  • 专用參数变量(Special Parameters)
    • Bash 提前定义的特殊变量
    • 用户不能改动其值

位置变量

  • 是一组特殊的内置变量

    • 跟在脚本名后面的用空格隔开的每一个字符串
    • 1表示第1个參数值。……,9 表示第9个參数值
    • 10表示第10个參数值,{11} 表示第11个參数值。 ……
  • 位置參数的用途
    • 从 shell 命令/脚本 的命令行接受參数
    • 在调用 shell 函数时为其传递參数

专用參数变量

  • 命令行參数相关

    1. $* 获取当前shell脚本全部传參的參数,将全部位置參量看成一个字符串(以空格间隔)。相当于""$1$2$3$4$5
    2. $@ 将每一个位置參量看成单独的字符串(以空格间隔)"$!" "$2" "...."
    3. "$*" 将全部位置參量看成一个字符串(以$IFS间隔)。
    4. "$@" 将每一个位置參量看成单独的字符串(以空格间隔)
    5. $0 命令行上输入的Shell程序名。
    6. $# 表示命令行上參数的个数。
  • 进程状态相关

    1. $? 表示上一条命令运行后的返回值
    2. $$ 当前进程的进程号
    3. $! 显示运行在后台的最后一个作业的 PID
    4. $_ 在此之前运行的命令或脚本的最后一个參数
  • 演示样例



位置參数和 shift 命令

  • shift [n]
  • 将位置參量列表依次左移n次。缺省为左移一次
  • 一旦位置參量列表被移动,最左端的那个參数就会从列表中删除
  • 经常与循环结构语句一起使用,以便遍历每一个位置參数

    1. #!/bin/sh
    2. # ScriptName: pp_shift.sh
    3. # To test Positional Parameters & Shift.
    4. echo "The script name is : $0"
    5. echo '$1'=$1,'$2'=$2,'$3'=$3,'$4'=$4 -- '$#'="$#"
    6. echo '$@': "$@"
    7. shift # 向左移动全部的位置參数1次
    8. echo '$1'=$1,'$2'=$2,'$3'=$3,'$4'=$4 -- '$#'="$#"
    9. echo '$@': "$@"
    10. shift 2 # 向左移动全部的位置參数2次
    11. echo '$1'=$1,'$2'=$2,'$3'=$3,'$4'=$4 -- '$#'="$#"
    12. echo '$@': "$@"
    13. $ ./pp_shift.sh 1 b 3 d 4 f

退出/返回状态

$?:返回上一条语句或脚本运行的状态

  • 0:成功
  • 1-255:不成功
  • 1:运行正确
  • 126:命令或脚本没有运行权限
  • 127:命令没有找到

exit 命令

  • exit 命令用于退出脚本或当前Shell
  • exit n
    • n 是一个从 0 到 255 的整数
    • 0 表示成功退出,非零表示遇到某种失败
  • 返回值 被保存在状态变量 $? 中

演示样例

  1. $ echo $$ # 显示当前进程的 PID
  2. 9245
  3. $ echo $? # 显示在此之前运行的命令的返回值
  4. 0
  5. $ bash # 调用子Shell
  6. $ echo $$ # 显示当前进程的 PID
  7. 9474
  8. $ exit 1 # 指定返回值并返回父Shell
  9. $ echo $? # 显示上一个Shell/脚本的返回值
  10. 1
  11. $ list # 运行不存在的命令
  12. bash: list: command not found
  13. $ echo $?
  14. 127
  15. $ touch bbb.sh
  16. $ ./bbb.sh # 运行不具有运行权限的命令
  17. bash: ./bbb.sh: Permission denied
  18. $ echo $?
  19. 126

read

  • 从键盘输入内容为变量赋值

    • read [-p “信息”] [var1 var2 …]
    • 若省略变量名,则将输入的内容存入REPLY变量
  • 结合不同的引號为变量赋值
    • 双引號 ” ”:同意通过$符号引用其它变量值
    • 单引號 ’ ’:禁止引用其它变量值。$视为普通字符
    • 反撇号 :将命令运行的结果输出给变量

演示样例

  1. #!/bin/bash
  2. # This script is to test the usage of read
  3. # Scriptname: ex4read.sh
  4. echo "=== examples for testing read ==="
  5. echo -e "What is your name?
  6. \c"
  7. read name
  8. echo "Hello $name"
  9. echo
  10. echo -n "Where do you work? "
  11. read
  12. echo "I guess $REPLY keeps you busy!"
  13. echo
  14. read -p "Enter your job title: "
  15. echo "I thought you might be an $REPLY."
  16. echo
  17. echo "=== End of the script ==="

仅仅读变量

  • 是指不能被清除或又一次赋值的变量
  • readonly variable

演示样例

  1. [lrj@centos1 ~]$ myname=Osmond
  2. [lrj@centos1 ~]$ echo $myname
  3. Osmond
  4. [lrj@centos1 ~]$ readonly myname
  5. [lrj@centos1 ~]$ unset myname
  6. -bash: unset: myname: cannot unset: readonly variable
  7. [lrj@centos1 ~]$ myname="Osmond Liang"
  8. -bash: myname: readonly variable

同一时候输出多行信息

  • 使用 echo

  • 使用 here file

整数运算

  • Bash 变量没有严格的类型定义

    • 本质上 Bash 变量都是字符串
  • 若一个字面常量或变量的值是纯数字的。不包括字母或其它字符, Bash能够将其视为长整型值,并可做算数运算和比較运算。
  • Bash 也同意显式地声明整型变量
    • declare -i 变量名

算数运算符

算术运算扩展—在此没有空格的问题

  1. $ var=test
  2. $ echo $var
  3. test
  4. $ echo $varAA
  5. $varAA
  6. $ echo ${var}AA
  7. testAA
  8. $ ls
  9. a b c
  10. $ echo $(ls)
  11. a b c
  12. $ echo `ls`
  13. a b c

Shell内置命令

let

  • let 内置命令用于算术运算

    1. num2=1; echo $num2
    2. let num2=4+1; echo $num2
    3. let num2=$num2+1; echo $num2
  • 赋值符号和运算符两边不能留空格!
  • 假设将字符串赋值给一个整型变量时。则变量的值为 0
  • 假设变量的值是字符串,则进行算术运算时设为 0

    1. let num2=4 + 1
    2. let "num2=4 + 1" # 用引號忽略空格的特殊含义
  • 用 let 命令进行算术运算时。最好加双引號

expr

  • 通用的表达式计算命令
  • 表达式中參数与操作符必须以空格分开。
  • 表达式中的运算能够是算术运算,比較运算,字符串运算和逻辑运算。

    1. expr 5 % 3
    2. expr 5 \* 3 # 乘法符号必须被转义
    3. expr 2 + 5 \* 2 - 3 % 2
    4. expr \( 2 + 5 \) \* 2 3 # 括号必须被转义

浮点数运算

  • bash 仅仅支持整数运算
  • 能够通过使用 bc 或 awk 工具来处理浮点数运算

    1. n=$(echo "scale=3; 13/2" | bc )
    2. echo $n
    3. m=`awk 'BEGIN{x=2.45;y=3.123;printf "%.3f\n", x*y}'`
    4. echo $m

printf

  • printf 可用来按指定的格式输出变量
  • printf format 输出參数列表



演示样例

数组变量

  • Bash 2.x 以上支持一维数组,下标从 0 開始
  • 使用 declare 声明或直接给变量名加下标来赋值

    1. declare -a variable
    2. variable=(item1 item2 item2 ... )
  • 数组的引用

    1. ${variable[n]}

演示样例

declare

  • 内置命令 declare 可用来声明变量
  • declare [选项] variable[=value]

演示样例

  1. declare myname=osmond
  2. declare r myname=osmond
  3. unset myname
  4. declare myname=“Osmond Liang"
  5. declare –x myname2=lrj
  6. myname2=lrj
  7. declare –x myname2

变量及相关命令小结






条件測试

  • 条件測试能够推断某个特定条件是否满足

    • 測试之后一般会依据不同的測试值选择运行不同任务
  • 条件測试的种类
    • 命令成功或失败
    • 表达式为真或假
  • 条件測试的值
    • Bash中没有布尔类型变量

      • 退出状态为 0 表示命令成功或表达式为真
      • 非0 则表示命令失败或表达式为假
    • 状态变量 $? 中保存了退出状态的值

演示样例

条件測试语句

  • 语句

    • 格式1: test <測试表达式>
    • 格式2: [ <測试表达式> ]
    • 格式3: [[ <測试表达式> ]] (bash 2.x 版本号以上)
  • 说明
    • 格式1 和 格式2 是等价的,格式3是扩展的 test 命令
    • 在 [[ ]] 中能够使用通配符进行模式匹配
    • &&, ||, <, 和>能够正常存在于[[ ]]中。但不能在 [] 中出现
    • [和[[之后的字符必须为空格,]和]]之前的字符必须为空格
    • 要对整数进行关系运算也能够使用 (()) 进行測试

条件測试操作符

  • 条件測试表达式中可用的操作符

    • 文件測试操作符
    • 字符串測试操作符
    • 整数二元比較操作符
    • 使用逻辑运算符

文件測试

  • 測试:文件是否存在。文件属性,訪问权限等

字符串測试

  • 字符串按从左到右相应字符的ASCII码进行比較



字符串空值检查

检查空值

检查非空值

整数測试

  • 操作符两边必须留空格!

  • 操作符两边的空格可省略

演示样例



逻辑測试

演示样例


流程控制——分支

  • 流程控制语句

    • 分支

      • if 条件语句
      • case 选择语句
    • 循环
      • for 循环语句
      • while 循环语句
      • until 循环语句
      • select 循环与菜单
    • 循环控制
      • break 语句
      • continue 语句
    • 位置參数处理
      • shift 命令
      • getopts 命令

if 语句

  1. if expr1 # 假设 expr1 为真(返回值为0)
  2. then # 那么
  3. commands1 # 运行语句块 commands1
  4. elif expr2 # 若 expr1 不真。而 expr2 为真
  5. then # 那么
  6. commands2 # 运行语句块 commands2
  7. ... ... # 能够有多个 elif 语句
  8. else # else 最多仅仅能有一个
  9. commands4 # 运行语句块 commands4
  10. fi # if 语句必须以单词 fi 终止
  • elif 能够有随意多个(0 个或多个)
  • else 最多仅仅能有一个(0 个或 1 个)
  • if 语句必须以 fi 表示结束
  • exprX 通常为条件測试表达式;也能够是多个命令。以最后一个命令的退出状态为条件值。
  • commands 为可运行语句块,假设为空,需使用 shell 提供的空命令 “ : ”,即冒号。该命令不做不论什么事情,仅仅返回一个退出状态 0
  • if 语句能够嵌套使用

演示样例











case语句

  1. case expr in # expr 为表达式。关键词 in 不要忘!
  2. pattern1) # 若 expr 与 pattern1 匹配。注意括号
  3. commands1 # 运行语句块 commands1
  4. ;; # 跳出 case 结构
  5. pattern2) # 若 expr 与 pattern2 匹配
  6. commands2 # 运行语句块 commands2
  7. ;; # 跳出 case 结构
  8. ... ... # 能够有随意多个模式匹配
  9. *) # 若 expr 与上面的模式都不匹配
  10. commands # 运行语句块 commands
  11. ;; # 跳出 case 结构
  12. esac # case 语句必须以 esac 终止
  • 表达式 expr 按顺序匹配每一个模式,一旦有一个模式匹配成功,则运行该模式后面的全部命令,然后退出 case。
  • 假设 expr 没有找到匹配的模式。则运行缺省值 “ ) ” 后面的命令块 ( 相似于 if 中的 else );“ ) ” 能够不出现。
  • 所给的匹配模式 pattern 中能够含有通配符和“ | ”。

  • 每一个命令块的最后必须有一个双分号。能够独占一行,或放在最后一个命令的后面。








流程控制——循环

  1. for variable in list
  2. # 每一次循环。依次把列表 list 中的一个值赋给循环变量
  3. do # 循环体開始的标志
  4. commands # 循环变量每取一次值,循环体就运行一遍
  5. done # 循环结束的标志。返回循环顶部
  • 列表 list 能够是命令替换、变量名替换、字符串和文件名称列表 ( 可包括通配符 )。每一个列表项以空格间隔
  • for 循环运行的次数取决于列表 list 中单词的个数
  • 能够省略 in list 。省略时相当于 in “$@”

  • 首先将 list 的 item1 赋给 variable

    • 运行do和done之间的 commands
  • 然后再将 list 的 item2 赋给 variable
    • 运行do和done之间的 commands

      *如此循环,直到 list 中的全部 item 值都已经用完

演示样例

















break 和 continue

  • break [n]

    • 用于强行退出当前循环
    • 假设是嵌套循环。则 break 命令后面能够跟一数字 n。表示退出第 n 重循环(最里面的为第一重循环)
  • continue [n]
    • 用于忽略本次循环的剩余部分。回到循环的顶部,继续下一次循环
    • 假设是嵌套循环,continue 命令后面也可跟一数字 n,表示回到第 n 重循环的顶部

演示样例



for循环(C语言型)语法

  1. for ((expr1;expr2;expr3)) # 运行 expr1
  2. do # 若 expr2的值为真时进入循环,否则退出 for循环
  3. commands # 运行循环体,之后运行 expr3
  4. done # 循环结束的标志,返回循环顶部
  • 通常 expr1和 expr3是算数表达式; expr2是逻辑表达式
  • expr1 仅在循环開始之初运行一次
  • expr2 在每次运行循环体之前运行一次
  • expr3 在每次运行循环体之后运行一次

  • 首先运行 expr1

  • 运行 expr2
    • 其值为假时,终止循环
    • 其值为真时。运行do和done之间的 commands
    • 运行expr3,进入下一次循环

演示样例





while 循环语句

  1. while expr # 运行 expr
  2. do # 若expr的退出状态为0。进入循环,否则退出while
  3. commands # 循环体
  4. done # 循环结束标志,返回循环顶部

演示样例



* While 和输入重定向



* 使用管道为 while 传递输入



until 循环语句

  1. until expr # 运行 expr
  2. do # 若expr的退出状态非0。进入循环,否则退出until
  3. commands # 循环体
  4. done # 循环结束标志,返回循环顶部

演示样例







  • 将循环结果通过管道传递给其它命令处理(done |)

  • 后台运行循环(done &)

循环与菜单

  • 一般地。使用 while 循环配合 case实现
  • Bash 提供了专门的 select 循环
    • select 循环主要用于创建菜单
    • select 是个无限循环
      • 通常要配合 case 语句处理不同的选单及退出
      • select 循环的退出
        • 按 ctrl+c 退出循环
        • 在循环体内用 break 命令退出循环
        • 或用 exit 命令终止脚本

演示样例

  • 使用while循环实现菜单

循环结构——select 语法

  1. select variable in list
  2. do # 循环開始的标志
  3. commands # 循环变量每取一次值,循环体就运行一遍
  4. done # 循环结束的标志
  • 按数值顺序排列的菜单项(list item)会显示到标准输出
  • 菜单项的间隔符由环境变量 IFS 决定
  • 用于引导用户输入的提示信息存放在环境变量 PS3 中
  • 用户输入的值会被存储在内置变量 RELAY 中
  • 用户直接输入回车将又一次显示菜单
  • 与 for 循环相似。省略 in list 时等价于 in “$*”

演示样例






位置參数和命令行參数处理

  • 在脚本中经常使用流程控制处理位置參数

    • 循环结构:while、for
    • 多分支结构:case
  • 在脚本中经常使用例如以下命令配合位置參数处理
    • shift
    • getopts

演示样例

  • 參数位置遍历









  • 位置參数处理举例

选项和參数

  • mybackup -z -c /etc/mybackup.conf -r -v ./foo.txt ./mydir
  • -z是个选项(option)。以减号開始的单字符
  • -c也是个选项,/etc/mybackup.conf 是该选项的附加參数(additional argument )
  • -r和-v也是选项,且不带附加參数
  • ./foo.txt 和 ./mydir 是脚本的处理对象,他们是不与不论什么选项相关的參数,在POSIX®标准中称其为“操作 对象/数”(operands)

  • 依照Linux的命令行书写规范。例如以下命令行

    • mybackup -z -c /etc/mybackup.conf -r -v ./foo.txt ./mydir
  • 也能够写成例如以下的等价形式

    • mybackup -zr -c /etc/mybackup.conf -v ./foo.txt ./mydir
    • mybackup -zv -c /etc/mybackup.conf -r ./foo.txt ./mydir
    • mybackup -vr -c /etc/mybackup.conf -z ./foo.txt ./mydir
    • mybackup -vz -c /etc/mybackup.conf -r ./foo.txt ./mydir
    • mybackup -zrv -c /etc/mybackup.conf ./foo.txt ./mydir

    - mybackup -zrvc /etc/mybackup.conf ./foo.txt ./mydir

  • 用户使用自己的代码分析这些选项将变得十分困难
  • Shell的内置命令getopts能够识别全部常见的选项格式,为用户处理选项和參数提供了方便

内置命令——getopts

  • getopts OPTSTRING VARNAME [ARGS…]
  • OPTSTRING
    • 是由若干有效的选项标识符组成的选项字符串
    • 若某选项标识符后有冒号,则表示此选项有附加參数
    • 若整个字符串前有冒号,将使用“安静”的错误模式
  • VARNAME :每次匹配成功的选项保存在变量中
  • ARGS : 參数列表。省略时为 ”$@”

  • getopts c:zrv opt

  • getopts :c:zrv opt

  • getopts的运行过程

    • 通常须要以循环的方式运行多次 getopts 来解析位置參数中的选项以及可能存在的选项附加參数

      *每次调用 getopts,将会处理參数列表中的“下一个”选项

      • 将选项存储在VARNAME变量中
      • 将此选项相应的附加參数存储在环境变量OPTARG中
      • 对环境变量OPTIND进行自增操作,使 $OPTIND 总是指向原始參数列表中“下一个”要处理的元素位置
      • 若VARNAME与OPTSTRING的全部选项均不匹配,则做“invalid option”的错误设置
      • 若某选项的參数不存在。则做“required argument not found”的错误设置
  • getopts的错误报告模式

  • 冗余(Verbose)模式( OPTSTRING 不以:开头)
    • “invalid option”

      • VARNAME=“?” ; unset OPTARG
    • “required argument not found”
      • VARNAME=“?” ; unset OPTARG 并输出错误信息
  • 安静(Silent)模式( OPTSTRING 以:开头)
    • “invalid option”

      • VARNAME=“?” ; OPTARG=‘无效的选项字符’
    • “required argument not found”
      • VARNAME=“:” ; OPTARG=‘与參数相应的选项字符’

while循环与getopts处理

  1. while getopts OPTSTRING VARNAME
  2. do
  3. case $VARNAME in
  4. …) ………… ;;
  5. …) ………… ;;
  6. :) ………… ;;
  7. \?) ………… ;;
  8. esac
  9. done
  • getopts 返回假时终止 while 循环

    • 当 getopts 遭遇到第一个非选项參数时终止解析
    • 当 getopts 遭遇到 “–”參数时终止解析

getopts的注意事项

  • getopts 不能解析 GNU-style 长參数 (–myoption)
  • getopts从不改变原始位置參数
    • 若希望移动位置參数,需手工运行 shift
  • getopts会自己主动对变量 OPTIND 做自增处理
    • OPTIDX的初始值为 1
    • 若要又一次解析命令行參数,需将OPTIDX的值置为 1
  • getopts 遭遇到第一个非选项參数时终止解析
    • 终止解析后运行命令

      • shift ((OPTIND-1))
    • 能够使 ”$@” 仅仅包括“操作 对象/数”(operands

演示样例

  • 演示样例1

    1. #!/bin/bash
    2. ## filename : pp_parse_getopts_1.sh
    3. while getopts "abc:def:ghi" flag
    4. do
    5. echo "$flag" $OPTIND $OPTARG
    6. done
    7. echo "Resetting"
    8. OPTIND=1
    9. while getopts "bc:def:ghi" flag
    10. do
    11. echo "$flag" $OPTIND $OPTARG
    12. done
    13. $ ./pp_parse_getopts_1.sh -a -bc foo -f "foo bar" -h gde
    14. $ ./pp_parse_getopts_1.sh -abf "foo bar" -h -gde c
    15. $ ./pp_parse_getopts_1.sh -abf foo bar -h c -gde
  • 演示样例2

    1. #!/bin/bash
    2. ## filename : pp_parse_getopts_2.sh
    3. while getopts ":abc:def:ghi" flag
    4. do
    5. echo "$flag" $OPTIND $OPTARG
    6. done
    7. echo "Resetting"
    8. OPTIND=1
    9. while getopts ":bc:def:ghi" flag
    10. do
    11. echo "$flag" $OPTIND $OPTARG
    12. done
    13. $ ./pp_parse_getopts_2.sh -a -bc foo -f "foo bar" -h -gde
    14. $ ./pp_parse_getopts_2.sh -abf "foo bar" -h -gde c
    15. $ ./pp_parse_getopts_1.sh -abf foo bar -h c -gde

* 演示样例3

  1. #!/bin/bash
  2. ## filename : mybackup_getopts.sh
  3. while getopts :zc:x:rv opt
  4. do
  5. case $opt in
  6. c) ConfFile=$OPTARG ;;
  7. x) ExcludeFile=$OPTARG ;;
  8. z) Compress=true ;;
  9. r) Recursive=true ;;
  10. v) Verbose=true ;;
  11. :)
  12. echo "$0: Must supply an argument to -$OPTARG." >&2
  13. exit 1
  14. ;;
  15. \?) echo "Invalid option -$OPTARG ignored." >&2 ;;
  16. esac
  17. done
  18. shift $((OPTIND-1)) ; echo $0 ; echo "$@"
  • 演示样例4

    1. #!/bin/bash
    2. ## filename : mybackup_getopts2.sh
    3. while getopts :zc:x:rv opt
    4. do
    5. case $opt in
    6. c) if [[ $OPTARG = -* ]]; then ((OPTIND--)) ; continue ; fi
    7. ConfFile=$OPTARG ;;
    8. x) ExcludeFile=$OPTARG ;;
    9. z) Compress=true ;;
    10. r) Recursive=true ;;
    11. v) Verbose=true ;;
    12. :)
    13. echo "$0: Must supply an argument to -$OPTARG." >&2
    14. exit 1
    15. ;;
    16. \?) echo "Invalid option -$OPTARG ignored." >&2 ;;
    17. esac
    18. done
    19. shift ((OPTIND-1)) ; echo $0 ; echo "$@"

函数

  • 为了避免大型脚本变得复杂、晦涩而使用函数
  • 将大型脚本代码切割成小块。将这些被命名的代码块称为函数
    • 一个函数就是一个子程序。用于完毕特定的任务

      • 如:加入一个用户、推断用户是否为管理员 等
  • 函数定义之后能够被使用它的主程序调用
    • 调用函数的方法与运行Shell命令无异
    • 能够在Shell脚本中调用(函数需先定义而后调用)
    • 在命令行上直接调用(定义函数的文件需先载入

合理使用Shell函数

  • 简化程序代码,实现代码重用

    • 实现一次定义多次调用。如:is_root_user()函数能够由不同的shell脚本反复使用。
  • 实现结构化编程
    • 使脚本内容更加简洁。增强程序的易读性
  • 提高运行效率
    • 将经常使用的功能定义为多个函数并将其保存在一个文件里

      • 相似其它语言的“模块”文件
    • 在 ~/bashrc 或命令行上使用 source 命令调用这个文件
    • 此文件里定义的多个函数一次性地调入内存,从而加快运行速度

函数的定义和调用

函数定义

函数调用

  • 仅仅需输入函数名就可以调用函数

    • 函数名
    • 函数名 參数1 參数2 …
  • 函数必须在调用之前定义

函数的存储和显示

  • 函数的存储

    • 函数和调用它的主程序保存在同一个文件里

      *函数的定义必须出如今调用之前

      *函数和调用它的主程序保存在不同的文件里

      • 保存函数的文件必须先使用 source 命令运行,之后才干调用当中的函数
  • 函数的显示
    • 显示当前Shell可见的全部函数名

      • $ declare -F
    • 显示当前Shell可见的全部(指定)的函数定义
      • $ declare -f
      • $ declare -f

演示样例

  • 函数的定义和调用

    • 演示样例1

      1. #!/bin/bash
      2. ## filename: all_in_one_backup_select.sh
      3. ### User define Function (UDF) ###
      4. sql_bak () { echo "Running mysqldump tool..."; }
      5. sync_bak () { echo "Running rsync tool..."; }
      6. git_bak () { echo "Running gistore tool..."; }
      7. tar_bak () { echo "Running tar tool..."; }
      8. ### Main script starts here ###
      9. PS3="Please choose a backup tools : "
      10. select s in mysqldump rsync gistore tar quit ; do
      11. case $REPLY in
      12. 1) sql_bak ;;
      13. 2) sync_bak ;;
      14. 3) git_bak ;;
      15. 4) tar_bak ;;
      16. 5) exit ;;
      17. esac
      18. done
    • 演示样例2

      1. #!/bin/bash
      2. ## filename: /root/bin/my_backup_functions.sh
      3. ### User define Function (UDF) ###
      4. sql_bak () { echo "Running mysqldump tool..."; }
      5. sync_bak () { echo "Running rsync tool..."; }
      6. git_bak () { echo "Running gistore tool..."; }
      7. tar_bak () { echo "Running tar tool..."; }
    • 演示样例3

      1. #!/bin/bash
      2. ## filename: all_in_one_backup_select.sourcefunc.sh
      3. source /root/bin/my_backup_functions.sh
      4. ### Main script starts here ###
      5. PS3="Please choose a backup tools : "
      6. select s in mysqldump rsync gistore tar quit ; do
      7. case $REPLY in
      8. 1|[mM]ysqldump) sql_bak ;;
      9. 2|[rR]sync) sync_bak ;;
      10. 3|[gG]istore) git_bak ;;
      11. 4|[tT]ar) tar_bak ;;
      12. 5) exit ;;
      13. esac
      14. done

函数与变量

  • 參数(Arguments)

    • 调用函数时。使用位置參数的形式为函数传递參数
    • 函数内的1−{n} 、∗和@ 表示其接收的參数
    • 函数调用结束后位置參数 1−{n} 、∗和@ 将被重置为调用函数之前的值
    • 在主程序和函数中,$0始终代表脚本名
  • 变量(Variables)

    *函数内使用 local 声明的变量是局部(Local)变量

    *局部变量的作用域是当前函数以及其调用的全部函数

    • 函数内未使用 local 声明的变量是全局(Global)变量

      *即主程序和函数中的同名变量是一个变量(地址一致

演示样例

  • 函数与位置參数

    • 演示样例1

      1. #!/bin/bash
      2. ## filename: pp_and_function.sh
      3. echo "===Print positional parameters in main :"
      4. echo "$0: $*"
      5. pp1(){
      6. echo 'f1--Print $* parameters in fun1 :' ; echo "$0: $*"
      7. }
      8. pp2(){
      9. echo 'f2--Print $* parameters in fun1 :' ; echo "$0: $*"
      10. pp1 1st 2nd 3th 4th 5th 6th 7th 8th 9th
      11. echo 'f2--Print $* parameters in fun1 :' ; echo "$0: $*"
      12. }
      13. pp1 1 2 3 4 5 6 7 8 9
      14. echo "===Print positional parameters in main :"
      15. echo "$0: $*"
      16. pp2 I II III IV V VI VII VIII IX
      17. ./pp_and_function.sh a b c d e f g h i
    • 演示样例2

      1. #!/bin/bash
      2. ## filename: function_max.sh
      3. # User define Function (UDF)
      4. usage () {
      5. echo "List the MAX of the positive integers in command line. "
      6. echo "Usage: `basename $0` <num1> <num2> [ <num3> ... ]"
      7. exit
      8. }
      9. max () {
      10. [[ -z $1 || -z $2 ]] && usage
      11. largest=0
      12. for i ; do ((i>largest)) && largest=$i ; done
      13. }
      14. ### Main script starts here ###
      15. max "$@"
      16. echo "The largest of the numbers is $largest."
      17. ./function_max.sh 1 58 111 32768 66
  • 由于largest变量在函数max内没有使用local声明,所以它是全局的

函数的结束与返回值

  • 当函数的最后一条命令运行结束函数即结束

    • 函数的返回值就是最后一条命令的退出码
    • 其返回值被保存在系统变量$?中
  • 能够使用 return 或 exit 显式地结束函数
    • return [N]

      • return 将结束函数的运行
      • 能够使用 N 指定函数返回值
    • exit [N]
      • exit 将中断当前函数及当前Shell的运行
      • 能够使用 N 指定返回值

演示样例

  1. #!/bin/bash
  2. ## filename: function_max2.sh
  3. # User define Function (UDF)
  4. max2 () {
  5. if [[ -z $1 || -z $2 ]] ; then
  6. echo "Need 2 parameters to the function." ; exit
  7. fi
  8. [ $1 -eq $2 ] &&
  9. { echo "The two numbers are equal." ; exit ; }
  10. (($1>$2)) && return $1 || return $2
  11. }
  12. ### Main script starts here ###
  13. read -p "Please input two integer numbers : " n1 n2
  14. echo "n1=$n1 , n2=$n2“
  15. max2 $n1 $n2
  16. return_val=$?
  17. echo "The larger of the two numbers is $return_val."

函数返回值

  • 使用全局变量引用函数的值不利于结构化编程
  • 使用 return 或 exit 仅仅能返回整数值
  • 使用标准输出实现函数的返回值
    • 是一种通用的方法。既能返回整数又能返回字符串
    • 函数结束前使用 echo 命令将结果显示到标准输出
    • 调用函数时使用例如以下的格式将函数的输出结果存到变量 RES 中,之后便可使用变量 $RES 的值(或输出、或运行測试、或进一步处理等)
    • RES=$(functionName)
    • echo $RES

演示样例

  • 使用标准输出返回函数值

    1. #!/bin/bash
    2. ## filename: function_to-upper.sh
    3. # User define Function (UDF)
    4. to_upper () {
    5. local str="$@"
    6. local output
    7. output=$(tr '[a-z]' '[A-Z]'<<<"${str}")
    8. echo $output
    9. }
    10. ### Main script starts here ###
    11. to_upper "This Is a TEST"
    12. res=$(to_upper "$@")
    13. echo "$res"
    14. res=$(to_upper "$1")
    15. [[ $res == "YES" ]] && echo "Continue..." || echo "Stop"
    16. ./function_to-upper.sh YES we are
    17. ./function_to-upper.sh No we are not
  • 系统INIT 启动脚本的结构——/etc/rc.d/init.d/*

linux&amp;shell的更多相关文章

  1. 详解Linux交互式shell脚本中创建对话框实例教程_linux服务器

    本教程我们通过实现来讲讲Linux交互式shell脚本中创建各种各样对话框,对话框在Linux中可以友好的提示操作者,感兴趣的朋友可以参考学习一下. 当你在终端环境下安装新的软件时,你可以经常看到信息 ...

  2. linux中shell变量$#,$@,$0,$1,$2的含义解释

    linux中shell变量$#,$@,$0,$1,$2的含义解释: 变量说明: $$ Shell本身的PID(ProcessID) $! Shell最后运行的后台Process的PID $? 最后运行 ...

  3. 使用C#给Linux写Shell脚本(下篇)

    在上篇的<使用C#给Linux写Shell脚本>结尾中,我们留下了一个关于C#如何调用BashShell的问题.在文章发布之后,我留意到有读者留言推荐使用“Pash”(一款类PowerSh ...

  4. linux中用shell获取昨天、明天或多天前的日期

    linux中用shell获取昨天.明天或多天前的日期 时间 -- :: BlogJava-专家区 原文 http://www.blogjava.net/xzclog/archive/2015/12/0 ...

  5. linux中用shell获取时间,日期

    linux中用shell获取昨天.明天或多天前的日期:在Linux中对man date -d 参数说的比较模糊,以下举例进一步说明:# -d, --date=STRING display time d ...

  6. Linux 的shell 字符串截取很有用。有八种方法。

    一 Linux 的字符串截取很有用.有八种方法. 假设有变量 var=http://www.linuxidc.com/123.htm 1  # 号截取,删除左边字符,保留右边字符. echo ${va ...

  7. Linux Bash Shell 快速入门

    BASH 的基本语法 最简单的例子 —— Hello World! 关于输入.输出和错误输出 BASH 中对变量的规定(与 C 语言的异同) BASH 中的基本流程控制语法 函数的使用 2.1     ...

  8. linux,shell输入反斜杠显示'W'。

    linux,shell输入反斜杠显示'W'. solution: 字体必须为"Courier New".

  9. 06 Linux下Shell介绍

    一.概述 每个人在成功登陆Linux后,系统会出现不同的提示符号,例如$,~,#等,然后你就可以开始输入需要的命令.若命令正确,系统就会依据命令的要求来执行,直到注销系统为止,在登陆到注销期间,输入的 ...

  10. linux BASH shell设置字体与背景颜色

    linux BASH shell下设置字体及背景颜色的方法. BASH shell下设置字体及背景颜色  echo -e "\e[31mtest\e[41m"  \e[30m 将字 ...

随机推荐

  1. 【2-SAT】【DFS】【分类讨论】Gym - 101617K - Unsatisfying

    题意:给你一张2-SAT,问你加至少几句a V b(不能用非运算)这样的语句,使得其无法全为真. 如果最开始没有左右两项都含非运算的析取表达式,则无解,因为显然你可以对每一项的不含非的那项规定为真,使 ...

  2. PHP -- 模拟测试上传文件

    FROM :http://web.itivy.com/article-740-1.html QQ上传文件为什么那么快? “QQ上传大文件/QQ群发送大文件时,可以在极短的时间内完成”是如何做到的. 有 ...

  3. CAShapeLayer实现音量大小动态改变

    我是效果图 实现如图这效果一般会直接通过多张图进行切换进行完成.但这样的处理,会浪费App的资源存储空间,而且效率也不高.那么今天我们用CAShapeLayer实现以下吧. 拆分: 1.一个椭圆 2. ...

  4. java同步ldap实例

    1.准备过程 1.1  在windows server 2008 R2操作系统下搭建AD域,不懂得看链接地址:http://99532720.blog.51cto.com/2194488/696611 ...

  5. Spring ListFactoryBean实例

    ListFactoryBean”类为开发者提供了一种在Spring的bean配置文件中创建一个具体的列表集合类(ArrayList和LinkedList). 这里有一个 ListFactoryBean ...

  6. 如何删除Struts2动作的后缀扩展名

    在Struts2中,所有动作类有一个默认的后缀 .action 扩展. 例如, <struts> <package name="default" namespac ...

  7. CDMA,GPRS,3G有什么区别

    1 CDMA: 我们常说的CDMA 是IS-95A CDMA的简称 ,属于第二代通信技术(2G)的一种,属于北美的技术.另一种技术是GSM,属于欧洲的技术.这两种实现的原理不同,各有各的优点: 2 G ...

  8. SharePoint Server 2013 Offline Installation (without Internet)

    转自:http://social.msdn.microsoft.com/Forums/sharepoint/zh-CN/08f90e0f-1f52-4eba-9f6e-4dd635ffaadc/sha ...

  9. Ext表格控件

    表格控件其实也就是帮我们完成了数据的填充工作而已,具体的数据源.要显示的列,列的定制.数据源中的哪条数据显示在哪个列中等属性还是需要我们自己手动配置的,所以我们分下面几步来完成网格控件的数据绑定: 1 ...

  10. Ant构建文件解析

    <?xml version="1.0" encoding="UTF-8"?> <!-- 在Ant脚本中,project是这个XML文档的根结点 ...