•  

    学习目标

    目标

    能够使用字符串的3种方式

    掌握Shell字符串拼接

    掌握shell字符串截取的常用格式

    能够定义Shell索引数组和关联数组

    能够使用内置命令alias,echo,read,exit,declare操作

    掌握Shell的运算符操作

     

     

    Shell字符串变量

    介绍

    字符串(String)就是一系列字符的组合。字符串是 Shell 编程中最常用的数据类型之一(除了数字和字符串,也没有其他类型了)

     

    字符串的3种格式

    1. 单引号方式
    2. 双引号方式, 推荐
    3. 不用引号方式

    image-20200610164651589

     

    字符串的3种格式区别

    1. 使用单引号 '' 的字符串:

      任何字符都会原样输出,在其中使用变量是无效的。

      image-20200610163234693

    2. 由双引号" "包围的字符串:

      其中包含了变量,那么该变量会被解析得到值,而不是原样输出。

      字符串中还可以出现双引号的子字符串,但是需要转义。

      image-20200610163716853

    3. 不被引号包围的字符串

      不被引号包围的字符串中出现变量时也会被解析,这一点和双引号" "包围的字符串一样。

      字符串中不能出现空格,否则空格后边的字符串会作为其他变量或者命令解析。

      image-20200610164851784

      image-20200610164953256

     

    获取字符串的长度

    语法

    含义

    获取字符串的字符长度

    演示

    image-20200610165428148

     

     

    字符串拼接方式

    1. 无符号拼接

    2. 双引号拼接

    3. 混合拼接

       

    演示

    image-20200610171003827

     

     

     

     

    字符串截取语法

    格式说明
    ${变量名:start:length}从 string 字符串的左边第 start 个字符开始,
    向右截取 length 个字符。
    ${变量名:start}从 string 字符串的左边第 start 个字符开始截取,直到最后。
    ${变量名:0-start:length}从 string 字符串的右边第 start 个字符开始,
    向右截取 length 个字符。
    ${变量名:0-start}从 string 字符串的右边第 start 个字符开始截取,直到最后。
    ${变量名#*chars}从 string 字符串第一次出现 *chars 的位置开始,
    截取 *chars 右边的所有字符。
    ${变量名##*chars}从 string 字符串最后一次出现 *chars 的位置开始,
    截取 *chars 右边的所有字符。
    ${变量名%chars*}从 string 字符串最后第一次出现 chars* 的位置开始,
    截取 chars* 左边的所有字符。
    ${变量名%%chars*}从 string 字符串第一次出现 chars* 的位置开始,
    截取 chars* 左边的所有字符

     

    截取演示

     

     

     

    Shell索引数组变量:定义-获取-拼接-删除

     

    介绍

    Shell 支持数组(Array), 数组是若干数据的集合,其中的每一份数据都称为数组的元素。

    注意Bash Shell 只支持一维数组,不支持多维数组。

     

    数组的定义

    语法

    在 Shell 中,用括号( )来表示数组,数组元素之间用空格来分隔. 语法为:

    注意,赋值号 = 两边不能有空格

    演示

    1.定义数字存储100,3,22,58,77,17,20

    2.Shell 是弱类型的,它并不要求所有数组元素的类型必须相同

    Shell数组元素定义后不是固定的, 定义后还可以赋值

    3.可以给指定元素赋值初始化

    由于上面只赋值了3个元素, 所以数组的长度是3

    image-20200629143058345

     

    数组的获取

    语法

    1.通过下标获取元素值,index从0开始

    注意使用{ }

    2.获取值同时复制给其他变量

    3.使用 @* 可以获取数组中的所有元素

    4.获取数组的长度或个数

    5.获取数组指定元素的字符程度

     

    演示

    image-20200629143255421

    image-20200629144217560

    image-20200629144245753

    image-20200629144359259

     

    数组的拼接

    介绍

    所谓 Shell 数组拼接(数组合并),就是将两个数组连接成一个数组

    语法

    使用 @* 获取数组所有元素之后进行拼接

    演示

    image-20200629145734269

     

    数组的删除

    介绍

    删除数组指定元素数据和删除整个数组数据

     

    语法

    删除数组指定元素数据

    删除整个数组

     

    演示

    image-20200629163807706

     

     

    Shell内置命令

    内置命令介绍

    Shell 内置命令,就是由 Bash Shell 自身提供的命令,而不是文件系统中的可执行文件。

    使用type 来确定一个命令是否是内置命令:

    image-20200630083936549

    通常来说,内置命令会比外部命令执行得更快,执行外部命令时不但会触发磁盘 I/O,还需要 fork 出一个单独的进程来执行,执行完成后再退出。而执行内置命令相当于调用当前 Shell 进程的一个函数, 还是一个进程, 减少了上下文切换。

    内置命令列表

    命令说明
    :扩展参数列表,执行重定向操作
    .读取并执行指定文件中的命令(在当前 shell 环境中)
    alias为指定命令定义一个别名
    bg将作业以后台模式运行
    bind将键盘序列绑定到一个 readline 函数或宏
    break退出 for、while、select 或 until 循环
    builtin执行指定的 shell 内建命令
    caller返回活动子函数调用的上下文
    cd将当前目录切换为指定的目录
    command执行指定的命令,无需进行通常的 shell 查找
    compgen为指定单词生成可能的补全匹配
    complete显示指定的单词是如何补全的
    compopt修改指定单词的补全选项
    continue继续执行 for、while、select 或 until 循环的下一次迭代
    declare声明一个变量或变量类型。
    dirs显示当前存储目录的列表
    disown从进程作业表中刪除指定的作业
    echo将指定字符串输出到 STDOUT
    enable启用或禁用指定的内建shell命令
    eval将指定的参数拼接成一个命令,然后执行该命令
    exec用指定命令替换 shell 进程
    exit强制 shell 以指定的退出状态码退出
    export设置子 shell 进程可用的变量
    fc从历史记录中选择命令列表
    fg将作业以前台模式运行
    getopts分析指定的位置参数
    hash查找并记住指定命令的全路径名
    help显示帮助文件
    history显示命令历史记录
    jobs列出活动作业
    kill向指定的进程 ID(PID) 发送一个系统信号
    let计算一个数学表达式中的每个参数
    local在函数中创建一个作用域受限的变量
    logout退出登录 shell
    mapfile从 STDIN 读取数据行,并将其加入索引数组
    popd从目录栈中删除记录
    printf使用格式化字符串显示文本
    pushd向目录栈添加一个目录
    pwd显示当前工作目录的路径名
    read从 STDIN 读取一行数据并将其赋给一个变量
    readarray从 STDIN 读取数据行并将其放入索引数组
    readonly从 STDIN 读取一行数据并将其赋给一个不可修改的变量
    return强制函数以某个值退出,这个值可以被调用脚本提取
    set设置并显示环境变量的值和 shell 属性
    shift将位置参数依次向下降一个位置
    shopt打开/关闭控制 shell 可选行为的变量值
    source读取并执行指定文件中的命令(在当前 shell 环境中)
    suspend暂停 Shell 的执行,直到收到一个 SIGCONT 信号
    test基于指定条件返回退出状态码 0 或 1
    times显示累计的用户和系统时间
    trap如果收到了指定的系统信号,执行指定的命令
    type显示指定的单词如果作为命令将会如何被解释
    typeset声明一个变量或变量类型。
    ulimit为系统用户设置指定的资源的上限
    umask为新建的文件和目录设置默认权限
    unalias刪除指定的别名
    unset刪除指定的环境变量或 shell 属性
    wait等待指定的进程完成,并返回退出状态码

    alisa内置命令

    介绍

    alisa 用于给命令创建别名。若该命令且不带任何参数,则显示当前 Shell 进程中的所有别名列表。

    image-20200630084814561

    上面是系统为了方便命令操作默认将部分命令创建为别名 ll 的命令与ls -l的效果一样, 就是因为ll是别名

     

    alias别名定义语法

     

    unalias 别名删除语法

    删除指定的别名

    删除当前Shell环境中所有的别名

    注意: 以上2种方式删除都是临时删除当前Shell的别名, 如果想永久删除必须去配置文件中手动删除

     

    alias演示

    输出所有进行信息

    给上面的命令起别名为psList, 并使用psList输出

    image-20200630091546806

    删除psList别名

    image-20200630091747276

     

    echo内置命令

    echo命令介绍

    echo 是一个 Shell 内置命令,用于在终端输出字符串,并在最后默认加上换行符

     

    echo输出语法

    默认输出换行语法

     

    输出不换行语法

    输出演示

    1.创建echo1.sh

     

    2.编辑文件内容并保存

    文件内容

    3.运行echo1.sh文件观察效果

    image-20200630094335603

    itheima shell输出为一行

    echo输出转义字符

    /n 转义字符

    用于echo输出字符串非结尾处的换行, 但是默认echo无法解析/n 转义字符

    演示

    image-20200630100606558

     

    -e 参数

    -e 参数用于解析转义字符

    演示

    image-20200630100653445

     

    /c 转义字符

    /c 用于强制清除echo的结尾换行输出

    演示

    image-20200630100953178

     

     

    read内置命令

    介绍

    read 是 Shell 内置命令,用于从标准输入中读取数据并赋值给变量。如果没有进行重定向,默认就是从终端控制台读取用户输入的数据;如果进行了重定向,那么可以从文件中读取数据。

    后续Shell重定向讲解

     

    语法

    options表示选项,如下表所示;var表示用来存储数据的变量,可以有一个,也可以有多个。

    optionsvar都是可选的,如果没有提供变量名,那么读取的数据将存放到环境变量 REPLY 中。

    options支持的参数

    选项说明
    -a array把读取的数据赋值给数组 array,从下标 0 开始。
    -d delimiter用字符串 delimiter 指定读取结束的位置,而不是一个换行符(读取到的数据不包括 delimiter)。
    -e在获取用户输入的时候,对功能键进行编码转换,不会直接显式功能键对应的字符。
    -n num读取 num 个字符,而不是整行字符。
    -p prompt显示提示信息,提示内容为 prompt。
    -r原样读取(Raw mode),不把反斜杠字符解释为转义字符。
    -s静默模式(Silent mode),不会在屏幕上显示输入的字符。当输入密码和其它确认信息的时候,这是很有必要的。
    -t seconds设置超时时间,单位为秒。如果用户没有在指定时间内输入完成,那么 read 将会返回一个非 0 的退出状态,表示读取失败。
    -u fd使用文件描述符 fd 作为输入源,而不是标准输入,类似于重定向。

     

    示例1:多个变量赋值

    需求

    使用 read 命令给多个变量赋值

    步骤

    1.创建文件read1.sh

    2.编辑文件, 编写read命令提示用户输入多个信息个多个变量赋值, 保存文件

    3.执行read1.sh文件

    演示

    创建文件

    编辑文件

     

    read1.sh文件内容

     

    运行效果

    image-20200630112030560

     

    示例2:读取一个字符

    需求

    从终端控制台只读取一个字符

    步骤

    1.创建文件read2.sh文件

    2.编辑文件,编写read命令, 使用-n 1 参数读取一个字符

    3.执行文件

    演示

    创建文件

    编辑文件

    read2.sh文件内容

    注意

    -n 1只读取一个字符。运行脚本后,只要输入的一个字符,立即读取结束,不用等待用户按下回车键。

    printf '\n'语句用于换行,否则 echo 的输出结果会和用户输入的内容位于同一行,不容易区分

    运行效果image-20200630113542863

     

    示例3:限制时间输入

    需求

    在终端控制台输入时, 设置指定时间内输入密码

     

    步骤

    1. 创建文件read3.sh
    2. 编辑文件内容编写, 使用read命令-t seconds 限制输入时间, 使用-s 静默模式输入密码
    3. 执行文件

    演示

    创建文件

    编辑文件

    read3.sh文件内容

    执行文件运行效果

    image-20200630115148158

     

     

    exit内置命令

    介绍

    exit 用于退出当前 Shell 进程结束运行,并返回一个退出状态;使用$?可以接收这个退出状态.

    exit 命令可以接受一个整数值作为参数,代表退出状态。如果不指定,默认状态值是 0。

    一般情况下,退出状态为 0 表示成功,退出状态为非 0 表示执行失败(出错)了。

     

    语法

    正确退出语法

     

    错误退出语法

     

    exit应用场景

    1.结束当前Shell进程

    2.当Shell 进程执行出错退出时,可以返回不同的状态值代表不同的错误

    比如打开一个文件时,可以返回1 表示文件不存在,2 表示文件没有读取权限,3 表示文件类型不对。

     

    示例:Shell脚本文件中使用exit退出

    需求

    编写Shell脚本使用exit 退出, 退出时返回一个非0数字状态值, 执行脚本文件并打印返回状态值

     

    步骤

    1.创建exit.sh文件

    2.编辑exit.sh文件, 使用exit 数字 退出结束当前Shell

    3.执行文件,打印返回状态值

     

    演示

    创建文件

    编辑文件

    exit.sh文件内容: 使用exit 数字 退出结束当前Shell

    运行效果

    image-20200630150541964

     

    declare内置命令

    介绍

    declare命令用于声明 shell 变量。可用来声明变量并设置变量的属性,也可用来显示shell函数。若不加上任何参数,则会显示全部的shell变量与函数(与执行set指令的效果相同)。

     

    declare命令作用

    1.declare设置变量的属性

    2.查看全部Shell变量与函数

    3.实现关联数组变量

    不同于普通数组, 关联数组的下标是字符串, 通过字符串下标操作数据

     

    语法

    1.declare设置变量的属性语法

    +/-  "-"可用来指定变量的属性,"+"则是取消变量所设的属性。

    a 设置为普通索引数组

    A 设置为key-value关联数组

    r  将变量设置为只读, 也可以使用readonly

    x  设置变量成为环境变量,也可以使用export

    i  设置为整型变量。

    f 设置为一个函数变量

    2.查看全部Shell变量与函数定义语法

    declare 后无参数, 查询全部Shell变量与函数定义

    -f  仅查询显示函数定义。

    -F 仅查询显示函数名字

    3.实现key-value关联数组变量语法

    关联数组也称为“键值对(key-value)”数组,键(key)也即字符串形式的数组下标,值(value)也即元素值。

    declare也可以用于定义普通数组, -a 参数创建普通或索引数组 -A 创建关联数组

    declare -a 关联数组变量名=(值1 值2 ...)

    declare -a 关联数组变量名=([0]=值1 [1]=值2 ...)

    获取指定key的值

    获取所有值

     

    示例1:设置变量属性

    需求

    操作一个变量属性, 设置为整型\ 取消整型\设置为只读等操作

    步骤

    1. 使用declare设置一个整型变量并赋值
    2. 取消变量的整型类型, 尝试设置其他类型值尝试
    3. 设置变量只读, 尝试赋值测试

    演示

    image-20200630164537518

    示例2:查看当前Shell所有函数名

    查看所有变量与所有函数

    image-20200701093735283

    查看所有函数与定义

    image-20200701094119327

    查询所有函数名列表

    image-20200701094553443

     

    示例3:实现普通索引数组

    需求

    使用declare定义一普通数组,并获取打印元素数据的值

    步骤

    1.创建declare1.sh文件

    2.编辑declare1.sh文件内容, 使用declare定义普通数组数据与打印

    2.执行文件

    演示

    创建文件

    编辑文件

    declare1.sh文件内容

    运行结果

    image-20200701101651788

    示例4:实现key-value关联数组

    需求

    使用declare定义关联数组并打印输出

    演示

    image-20200701105943576

    注意: 使用参数-A 才是关联数组 , 关联数组无法使用索引获取

     

    Shell运算符

    expr命令

    介绍

    expr 是 evaluate expressions 的缩写,译为“表达式求值”。Shell expr 是一个功能强大,并且比较复杂的命令,它除了可以实现整数计算,还可以结合一些选项对字符串进行处理,例如计算字符串长度、字符串比较、字符串匹配、字符串提取等, 后续讲解。

    语法

    计算语法

    注意: 运算表达式

    获取计算结果赋值给新变量语法

    演示

    image-20200701150021331

    注意: 运算符表达式中每个数字与符号之间要有空格

    算术运算符

    算术运算符介绍

    下表列出了常用的算术运算符,假定变量 a 为 1,变量 b 为 2:

    运算符说明举例
    +加法expr $a + $b 结果为 3
    -减法expr $a - $b 结果为 -1
    *乘法expr $a \* $b 结果为 2
    /除法expr $b / $a 结果为 2
    %取余expr $b % $a 结果为 0
    =赋值a=$b 将把变量 b 的值赋给 a

    四则运算中如果使用了(), 也需要转义 \( 1 + 1 \)

    算术运算符演示

    operation1.sh脚本代码

    image-20200701145757594

     

     

     

    比较运算符

    整数比较运算符

    语法

    下表列出了常用的比较运算符,假定变量 a 为 1,变量 b 为 2:

    运算符说明举例
    -eqequals 检测两个数是否相等,相等返回 0, 否则返回1。[ $a -eq $b ] 返回 1。
    -nenot equals检测两个数是否不相等,不相等返回 true。[ $a -ne $b ] 返回 0。
    -gtgreater than检测左边的数是否大于右边的,
    是返回0, 否则1
    [ $a -gt $b ] 返回 1。
    -ltlower than检测左边的数是否小于右边的,
    是返回0, 否则1
    [ $a -lt $b ] 返回 0。
    -gegreater equals检测左边的数是否大于等于右边的,
    是返回0, 否则1
    [ $a -ge $b ]返回 1。
    -lelower equals检测左边的数是否小于等于右边的,
    是返回0, 否则1
    [ $a -le $b ]返回 0。
    <检测左边的数是否小于右边的,
    是返回0, 否则1
    (($a<$b)) 返回0
    <=检测左边的数是否小于等于右边的,
    是返回0, 否则1
    (($a<=$b)) 返回0
    >检测左边的数是否大于右边的,
    是返回0, 否则1
    (($a>$b)) 返回1
    >=检测左边的数是否大于等于右边的,
    是返回0, 否则1
    (($a>=$b)) 返回1

    注意: 整数比较运算符只支持整数,不支持小数与字符串(字符串比较后续讲解),除非字符串的值是整数数字。

    每个命令都有返回值, 这个后面我们会讲解退出状态再具体说明, 返回0代表成功, 返回1代表失败

    演示

    operation2.sh脚本代码

    运行效果

    image-20200704084311714

     

    字符串比较运算符

    介绍

    可以比较2个变量, 变量的类型可以为数字(整数,小数)与字符串

    语法

    下表列出了常用的字符串运算符,假定变量 a 为 "abc",变量 b 为 "efg":

    字符串比较可以使用 [[]][] 2种方式

    运算符说明举例
    == 或 =相等。用于比较两个字符串或数字,相同则返回 0。可以使用=[ $a == $b ]返回1
    [ $a = $b ] 返回 1
    [[ $a == $b ]] 返回1
    [[ $a = $b ]] 返回1
    !=不相等。用于比较两个字符串或数字,不相同则返回 0。[ $a != $b ] 返回 0
    [[ $a != $b ]] 返回 0
    <小于, 用于比较两个字符串或数字, 小于返回0, 否则返回1[ $a \< $b ] 返回 0
    [[ $a < $b ]] 返回 0
    >大于, 用于比较两个字符串或数字, 大于返回0, 否则返回1[ $a \> $b ] 返回 1
    [[ $a > $b ]] 返回 1
    -z检测字符串长度是否为0,为0返回 true。[ -z $a ] 返回 1。
    -n检测字符串长度是否不为 0,不为 0 返回 true。[ -n "$a" ] 返回 0。
    |0,1|[a ] 返回 0。  

    字符串比较没有 <= 可以通过 [[ "a" < "b" && "a" = "b" ]]

    演示

    operation6.sh脚本代码

    运行效果

    image-20200704110751421

     

    [[]][] 的区别

    区别1: word splitting的发生

    区别说明

    [[]] 不会有word splitting发生

    [] 会有word splitting发生

    word splitting介绍

    会将含有空格字符串进行分拆分割后比较

    示例

    image-20200704092823470

    通过 $? 获取上一个命令的退出状态, 0代表成功, 1代表失败

    区别2: 转义字符

    区别说明

    [[]]< 不需要转义, 格式为 [[ 字符串1 < 字符串2 ]]

    [] 需要对 <,>等 转义 , 格式为 [ 字符串1 \< 字符串2 ]

    示例

    [] 执行效果

    image-20200704100701687

    [[]] 执行效果, 不需要转义执行结果正确

    image-20200704100843230

    小结

    1. 了解有哪些比较运算符

      整数比较运算符

      比较方式: [](())

      方式1: [ 整数1 options 整数2 ]

      options: -eq -gt -ge -lt -le -ne

      方式2: ((整数1 options 整数2))

      options: < <= > >=

      字符串,数字比较运算符

      介绍: 可以比较2个变量, 变量的类型可以为字符串, 数字(整数,小数)

      比较方式: [][[]]

      方式1: [ 变量1 options 整数2 ]

      options: <,>,==,=,!=

      <,> 需要转义

      比较含有空格的字符串, 需要使用""号括起来, 否则会分割拆分

      方式2: [[ 变量1 options 变量2]]

      options: <,>,==,=,!=

      <,> 不需要转义

    2. 推荐使用哪个比较运算符?

      [[ 表达式 ]] 即可以实现数字和字符串比较, 并且不需要转义, 不会分割

     

    布尔运算符

    介绍

    运算符说明举例
    !非运算,取反, 表达式为 true 则返回 false,
    否则返回 true。
    [ ! false ] 返回 true。
    -oor 或运算,有一个表达式为 true 则返回 true。[ 表达式1 -o 表达式2 ] 返回 true。
    -aand 与运算,两个表达式都为 true 才返回 true。[ 表达式1 -a 表达式2 ] 返回 false。

    注意布尔运算符只能放在[] 才有效

    以后常使用布尔运算符与test命令进行连接条件测试, 后续讲解

    演示

    operation4.sh脚本代码脚本代码

     

    运行效果

    image-20200701170127507

     

     

    逻辑运算符

    介绍

    运算符说明举例
    &&逻辑的 AND[[ 表达式1 && 表达式2 ]] 返回 false
    ||逻辑的 OR[[ 表达式1 || 表达式2 ]] 返回 true

    注意: 使用&&|| 的运算符必须放在 [[]](())中才有效, 否则报错

    -a-o 的运算符必须放在 [] 在才有效 或 test命令中

    !可以用在[],[[]]中, 不可以在(())

    演示

    operation5.sh脚本代码

    运行效果

    image-20200701172201560

     

    文件测试运算符

    介绍

    文件测试运算符用于检测文件的各种属性。

    属性检测描述如下:

    操作符说明举例
    -b file检测文件是否是块设备文件,如果是,则返回 true。[ -b $file ] 返回 false。
    -c file检测文件是否是字符设备文件,如果是,则返回 true。[ -c $file ] 返回 false。
    -d filedirectory, 检测文件是否是目录,如果是,则返回 true。[ -d $file ] 返回 false。
    -f filefile, 检测文件是否是普通文件(既不是目录,也不是设备文件)
    ,如果是,则返回 true。
    [ -f $file ] 返回 true。
    -g file检测文件是否设置了 SGID 位,如果是,则返回 true。[ -g $file ] 返回 false。
    -k file检测文件是否设置了粘着位(Sticky Bit),如果是,
    则返回 true。
    [ -k $file ] 返回 false。
    -p file检测文件是否是有名管道,如果是,则返回 true。[ -p $file ] 返回 false。
    -u file检测文件是否设置了 SUID 位,如果是,则返回 true。[ -u $file ] 返回 false。
    -r fileread,检测文件是否可读,如果是,则返回 true。[ -r $file ] 返回 true。
    -w filewrite,检测文件是否可写,如果是,则返回 true。[ -w $file ] 返回 true。
    -x fileexecute, 检测文件是否可执行,如果是,则返回 true。[ -x $file ] 返回 true。
    -s filesize, 检测文件是否为空(文件大小是否大于0)
    ,不为空返回 true。
    [ -s $file ] 返回 true。
    -e fileexists, 检测文件(包括目录)是否存在,如果是,
    则返回 true。
    [ -e $file ] 返回 true。
    file1 -nt file2new than(nt), file1是否比file2新[ file1 -nt file2 ]
    file1 -ot file2old than(ot), file1是否比file2旧[ file1 -ot file2 ]

    其他检查符:

    语法

    演示

    operation6.sh脚本代码

     

    运行效果

    image-20200703085454524

    operation1.sh文件不可执行, 因为没有可执行权限

    查看operation1.sh文件权限

    image-20200703084825357

    给operation1.sh添加执行权限

    image-20200703085602758

    重新执行operation6.sh

    image-20200703085719905