1 Shell 概述

1Linux 提供的 Shell 解析器有

Ubuntu 使用的是dash

2bash sh 的关系

3Centos 默认的解析器是 bash

2 Shell 脚本入门

1)脚本格式 (结尾不是必须以 .sh 结尾,只是为了区分脚本以#!/bin/bash 开头(指定解析器)2)第一个 Shell 脚本:helloworld.sh 1)需求:创建一个 Shell 脚本,输出 helloworld (2)案例实操:(3)脚本的常用执行方式 第一种:采用 bash sh+脚本的相对路径或绝对路径(不用赋予脚本+x 权限) sh+脚本的相对路径

sh+脚本的绝对路径

bash+脚本的相对路径

bash+脚本的绝对路径

第二种:采用输入脚本的绝对路径或相对路径执行脚本(必须具有可执行权限+x,之前不需要是因为执行的是bin下的bash脚本,而文件是参数,这里相当于单独执行

①首先要赋予 helloworld.sh 脚本的+x 权限

②执行脚本 相对路径

绝对路径

注意:第一种执行方法,本质是 bash 解析器帮你执行脚本,所以脚本本身不需要执行权限。第二种执行方法,本质是脚本需要自己执行,所以需要执行权限。 【了解】第三种:在脚本的路径前加上“.”或者 source①有以下脚本

②分别使用 shbash./ . 的方式来执行,结果如下:

原因: 前两种方式都是在当前 shell 中打开一个子 shell 来执行脚本内容,当脚本内容结束,则子 shell 关闭,回到父 shell 中。 第三种,也就是使用在脚本路径前加“.”或者 source 的方式,可以使脚本内容在当前shell 里执行,而无需打开子 shell!这也是为什么我们每次要修改完/etc/profile 文件以后,需要 source 一下的原因。 开子 shell 与不开子 shell 的区别就在于,环境变量的继承关系,如在子 shell 中设置的当前变量,父 shell 是不可见的。

3 章 变量

3.1 系统预定义变量

1)常用系统变量 $HOME、$PWD、$SHELL、$USER 等 命令env 可以查看当前系统全局变量2)案例实操 1)查看系统变量的值

(2)显示当前 Shell 所有变量set

3.2 自定义变量

这里有一个很重要的概念,外部shell的全局变量对于内部嵌套的子shell是共享的,但是局部变量子shell不可见。和java不太一样。即便是全局变量,子shell更改了它的值,也只是在子shell里面赋值,退出之后的父shell并没有变化。

1)基本语法 1)定义变量:变量名=变量值,注意,=号前后不能有空格 (2)撤销变量:unset 变量名(3)声明静态变量:readonly 变量,注意:不能 unset (4)声明全局变量:先声明并赋值一个局部变量,然后在外层shell 使用 export【变量名】,这个全局变量,只在当前会话有效2)变量定义规则 1)变量名称可以由字母、数字和下划线组成,但是不能以数字开头,环境变量名建议大写(2)等号两侧不能有空格 (3)在 bash 中,变量默认类型都是字符串类型,无法直接进行数值运算。 (4)变量的值如果有空格,需要使用双引号或单引号括起来。 3)案例实操 1)定义变量 A(2)给变量 A 重新赋值

(3)撤销变量 A

(4)声明静态的变量 B=2,不能 unset

(5)在 bash 中,变量默认类型都是字符串类型,无法直接进行数值运算

(6)变量的值如果有空格,需要使用双引号或单引号括起来

(7)可把变量提升为全局环境变量,可供其他 Shell 程序使用

helloworld.sh 文件中增加 echo $B

发现并没有打印输出变量 B 的值

3.3 特殊变量

3.3.1 $n

1)基本语法 $n (功能描述:n 为数字,$0 代表该脚本名称,$1-$9 代表第一到第九个参数,十以上的参数,十以上的参数需要用大括号包含,如${10}) 补充:单引号勾选的内容不会把变量符号 $ 识别为变量符号,而是识别为字符串。双引号可以正常识别2)案例实操

3.3.2 $#

1)基本语法 $# (功能描述:获取所有输入参数个数,常用于循环,判断参数的个数是否正确以及加强脚本的健壮性)2)案例实操

3.3.3 $*$@

1)基本语法 $* (功能描述:这个变量代表命令行中所有的参数,$*把所有的参数看成一个整体$@ (功能描述:这个变量也代表命令行中所有的参数,不过$@把每个参数区分对待,类似于数组2)案例实操

3.3.4 $

1)基本语法 $?功能描述最后一次执行的命令的返回状态。如果这个变量的值为 0,证明上一个命令正确执行;如果这个变量的值为非 0(具体是哪个数,由命令自己来决定),则证明上一个命令执行不正确了2)案例实操 判断 helloworld.sh 脚本是否正确执行

4 章 运算符

在此之前,想要计算表达式,可以借助 expr 命令,但是想要获取值相当于把输入的运算符当作一个参数,那么此前我们要求字符间不能有空格,现在就必须添加空格。但是像乘号 * 这样还作为通配符的符号,我们还需要加反斜杠转义,太麻烦了,所以特别增添运算符来帮助获取具体的值

想要把表达式的值传入别的表达式,只能通过反引号包裹,或者是: $($(expr xxxx))

1)基本语法“$((运算式))” 或 “$[运算式]” 2)案例实操: 计算(2+3)* 4 的值

5 章 条件判断

-z : 判断是否是0值

3)案例实操

6 章 流程控制(重点)

6.1 if 判断

1)基本语法

为了防止出现空指针异常,有时候通过字符串拼接的方式避免,会这么写 if [ ‘$1’x = ‘fanxy’x ]

组合判断,可以用多个中括号,或者采用 -a (and)在同一个中括号进行判断。

2)案例实操 输入一个数字,如果是 1,则输出 banzhang zhen shuai,如果是 2,则输出 cls zhen mei如果是其它,什么也不输出。

6.2 case 语句

1)基本语法

6.3 for 循环

双小括号在Linux里面就可以直接使用数学表达式了。

注意不被括号引起来,两者没有区别!

6.4 while 循环

这里其实想用高级语言的写法,不想使用靠 $ 符号来完成的赋值造成使用太多中括号的麻烦,可以选择通过使用: let sum+=i

即 let 命令

7 read 读取控制台输入

这里就不能使用 $1 $2 这种获取参数的方式了,这是执行脚本的参数。

8 章 函数

8.1 系统函数

想要在脚本中,执行外面的系统函数,需要通过 $() 进行表达式替换,直接写会被当成字符串

8.1.1 basename

8.1.2 dirname

但有时候想获取绝对路径的时候,我们当前执行命令的path是相对路径,我们为了保证能获取绝对路径,可以通过cd和pwd来达到效果 path=(cd $(dirname $0); pwd)

8.2 自定义函数

调用前必须先声明!shell逐行执行!

方括号内容代表可有可无!传入的形参不需要专门定义名字,使用 $” />

注意!!!! : return 数值 (0 – 255) 调用函数不需要加括号,空格隔开即可

想要返回范围外的数值,可以通过echo,然后获取返回值的时候,通过赋值的方式,当然让函数返回值需要通过 $() 包裹,进行表达式替换

9 章 正则表达式入门

正则表达式使用单个字符串来描述、匹配一系列符合某个语法规则的字符串。在很多文本编辑器里,正则表达式通常被用来检索、替换那些符合某个模式的文本。在 Linux 中,grep, sed,awk 等文本处理工具都支持通过正则表达式进行模式匹配。

9.1 常规匹配

9.2 常用特殊字符

^xxxxx$ : 代表只能匹配和xxxxx完全相同的内容行,并不是指定的开头结尾

所以 ^$ :匹配所有空行

而为了完成达成我们指定开头和结尾的目标,可以利用 * 和 . 结合

如匹配 a 开头 en 结尾: ^a.*en$

指定中间出现某字符,即利用 .* 组合包裹住 指定字符即可

[ ] 中括号可以不用 ,分割,但是分割可读性更好

不光加上转义字符,还需要加单引号!

为了使用拓展的正则规则,可以通过命令追加 -E 就可以使用如 {数字} 代表出现数字次数

10 章 文本处理工具

10.1 cut

列号分割,可以指定多个列号,用逗号隔开,也可以指定类似正则的范围,如

1到6 :1-63以内 :-34以上: 4-

10.2 awk

用管道的话就不需要指定文件名了

BEGIN 和 END 分别代表所有数据读取行之前执行,和所有数据读取之后执行

awk分割空格的时候不予处理,直接按第一个词作为索引1

11 章 综合应用案例

11.1 归档文件

#!/bin/bash# 判断参数个数是否为1if [ $# -ne 1 ]thenecho "参数个数错误!应该输入一个参数,作为归档目录名!"exitfi# 从参数中获取目录名称,这里多余的echo是为了空行if [ -d $1 ]thenechoelseechoecho "目录不存在!"echoexitfi# 获取日期和路径DIR_NAME=$(basename $1)DIR_PATH=$(cd $(dirname $1); pwd)# 获取当前日期DATE=$(date +%y%m%d)# 定义生成的归档文件名称FILE=archive_${DIR_NAME}_${DATE}.tar.gzDEST=/root/archive/$FILEecho "开始归档..."echotar -czf $DEST $DIR_PATH/$DIR_NAMEif [ $" />

然后我们通过 crontab -e进行编辑系统定时任务,即可完成我们脚本的自动化部署

11.2 发送消息

这里 grep-i :忽略大小写

grep -m : 最大匹配个数