什么是bash shell脚本编程?
答:Linux里面有多种shell,而CentOS和redhat的默认shell是bash shell。至于shell脚本,这个跟windows操作系统里面的批处理文件有点像(.bat的文件)。不知道大家还是否记得Linux的哲学思想吗?其中有那么两点点:由众多目的的单一应用程序组成:一个程序只做一件事,且做好;组合目的的单一的小程序完成复杂的任务。我觉得shell脚本编程就很好的体现了这个哲学思想。shell脚本利用shell的功能缩写的一个“程序”,这个程序是使用纯文本文件,将一些shell的语法与命令(内部命令和外部命令)写在里面,搭配正则表达式、管道命令与数据流重定向等功能,以达到我们所想要的处理目的。
shell脚本有什么用?
shell脚本的用途有:
自动化常用命令
执行系统管理和故障排除
创建简单的应用程序
处理文本或文件
编程基本概念
编程逻辑处理方式:
顺序执行
循环执行
选择执行
shell编程:过程式、解释执行
编程语言的基本结构:
数据存储:变量、数组
表达式: a + b
语句:if
shell脚本是包含一些命令或声明,并符合一定格式的文本文件
格式要求:首行shebang机制
#!/bin/bash
shell编程的良好习惯
养成良好的shell编程习惯是必须的,假如要问为什么?你可以想想。假如你刚入职一家公司,接手了工作,然后让你执行某些脚本或者让你修改。假如别人一点注释的信息都没有,你看到代码第一眼会不会傻眼?假如代码上百行或者上千行,假如你一点都不了解的情况下去修改,那不相当于自己重新又写了一次脚本?多么浪费时间和精力。即便是看自己写的脚本,也难道日后维护的时候连自己都忘记掉。所以,一个良好的编程习惯,是必须的。
如下图,一些示例:
标注一些脚本的作者信息,该脚本的版本,和脚本的作用,和一些代码的作用等等基础信息。
脚本调试
当我们写完一个脚本的时候,很多时候,都难免出点小错误。所以写完脚本,都需要检查一下脚本是否有语法错误之类。
bash -n script.sh
检测脚本中是否存在语法错误
如下图,假如脚本没有错误,执行该命令不会任何提示
假如脚本有语法错误的,看下面第一次执行bash -n.此时就会报错了
bash -x /path/to/some_script
调试脚本
如下图,当调试脚本的时候,他会把执行脚本的步骤都显示出来,如果哪里出现了错误,我们能及时发现哪里有问题并更正过来。
bash中的变量
什么是变量?
变量:命名的内存空间
变量数据存储方式:
字符:
数值:整型,浮点型
变量的作用
1、数据存储格式
2、参与的运算
3、表示的数据范围
编程语言分为强类型语言和弱类型语言
强类型:
定义变量时必须指定类型、参与运算必须符合类型要求;调用未声明变量会产生错误
如java,python
弱类型:
无须指定类型,默认均为字符型;参与运算会自动进行隐式类型转换;变量无须事先定义可直接调用
如:bash 不支持浮点数
变量命名法则
1、不能使程序中的保留字:例如if, for;
2、只能使用数字、字母及下划线,且不能以数字开头
3、见名知义:如求加法的结果,用SumPlus等容易看出来的
4、统一命名规则:驼峰命名法(如SumPlus,尽量不用纯小写或者纯大写以免跟系统的其他变量冲突)
变量的种类
根据变量的生效范围等标准:
本地变量:生效范围为当前shell进程;对当前shell之外的其它shell进程,包括当前shell的子shell进程均无效
环境变量:生效范围为当前shell进程及其子进程
局部变量:生效范围为当前shell进程中某代码片断(通常指函数)
位置变量:$1, $2, ...来表示,用于让脚本在脚本代码中调用通过命令行传递给它的参数
特殊变量:$?, $0, $*, $@, $#
下面通过示例来演示一下这些变量效果:
本地变量:(这个意思就像个人设置)
赋值格式 name='value'
value可以使用引用:
1.可以是直接字符窜;name="username"
2.变量引用:name"username"
3.命令引用name=`COMMAND`,name=$(COMMAND)
变量引用:$name,${name}
"" : 弱引用,其中的变量引用会被替换为变量值;
'' : 强引用,其中的变量引用不会被替换为变量值,而保持原字符串;
显示已定义的所有本地变量:set
销毁变量:unset name
可以看到alias里面这管理员用户和普通用户的别名里面的设置都是有点不一样的,这就是本地变量了
环境变量:(国有国法,所谓的国法,大家都要遵守,这就是可以理解为环境变量了)
变量声明、赋值: export name=VALUE 或 declare -x name=VALUE
变量引用:$name,$(name)
显示所有环境变量:export、env、printenv
销毁: unset name
bash有许多内建的环境变量:如PATH、HISTSIZE、PS1等等..
如下,大家使用外部命令的时候都同是通过PATH指定的环境变量去搜索命令的所在位置
位置变量:(在脚本里面的以$1、$2..$n,shift [n]换位置表示的变量)
赋值:script $1,$2,...对应调用第1、第2个等参数;
如下图,赋值$1为1,$2为10,计算1到10所有整数的和。
不过这个位置变量很有意思的是,等参数超过10个以后,第10个参数就会以$1(0)这样的参数形式出现,我暂时也没弄明白这是什么意思。如第一个参数是a,那么第到了第10个开始,第十个的变量显示出来的就是a0,第20个是b0,第30个是c0等等。。。
特殊变量:$?, $0, $*, $@, $#
$?,表示命令执行的返回状态,0表示成功,非0为失败,如下图所示
$0,表示命令本身
$#:传递给脚本有多少个参数,假如参数有两个,那么$#=2,如下图所示,假如$#的参数不等于2,那么该脚本就提示“请输出正确的地址参数”。
$*和$@,这两个,只有有“”双引号的时候才会显示有所不同的意思
$*:传递给脚本的所有参数
$@:引用传递给脚本的所有参数
如下图,当$*加上双引号以后,$*表示的是‘a b c’,这个是一个整体,但不加双引号,就是a和b和c
只读变量:只能声明,但不能修改和删除
赋值:readonlyname 、 declare -r name
bash中的算术运算
bash中的算术运算:(帮助文件见help let)
+(加), -(减), *(乘以), /(除以), %取模(取余), **(乘方)
实现算术运算:
(1) let var=算术表达式
(2) var=$[算术表达式]
(3) var=$((算术表达式))
(4) var=$(expr arg1 arg2 arg3 ...)
(5) declare –ivar= 数值
(6) echo ‘算术表达式’ | bc
乘法符号有些场景中需要转义,如*
bash有内建的随机数生成器:$RANDOM(1-32767)
echo $[$RANDOM%50] :0-49之间随机数, 如需要1-50之间的随机,则 $[$RANDOM%50]+1 即可
赋值
增强型赋值:
+=, -=, *=, /=, %=
let varOPERvalue
例如:let count+=3
自加3后自赋值
自增,自减:
let var+=1
let var++
let var-=1
let var--
逻辑运算
true, false
1, 0
与:
1 与 1 = 1
1 与 0 = 0
0 与 1 = 0
0 与 0 = 0
或:
1 或 1 = 1
1 或 0 = 1
0 或 1 = 1
0 或 0 = 0
非:!
! 1 = 0
! 0 = 1
短路运算:
短路与:
第一个为0,结果必定为0;
第一个为1,第二个必须要参与运算;
短路或:
第一个为1,结果必定为1;
第一个为0,第二个必须要参与运算;
异或:^
异或的两个值,相同为假,不同为真
聚集命令
有两种聚集命令的方法:
复合式:date; who | wc -l 命令会一个接一个地运行
子shell:(date; who | wc -l ) >>/tmp/trace,所有的输出都被发送给单个STDOUT和STDERR
bash退出状态码
进程使用退出状态来报告成功或失败
0 代表成功,1-255代表失败
$? 变量保存最近的命令退出状态
例如:
bash自定义退出状态码
exit [n]:自定义退出状态码;
注意:脚本中一旦遇到exit命令,脚本会立即终止;终止退出状态取决于exit命令后面的数字
注意:如果未给脚本指定退出状态码,整个脚本的退出状态码取决于脚本中执行的最后一条命令的状态码
条件测试
判断某需求是否满足,需要由测试机制来实现;
专用的测试表达式需要由测试命令辅助完成测试过程;
评估布尔声明,以便用在条件性执行中
若真,则返回0
若假,则返回1
测试命令:
test EXPRESSION
[ EXPRESSION ]
` EXPRESSION `
注意:EXPRESSION前后必须有空白字符
根据退出状态而定,命令可以有条件地运行
&& 代表条件性的AND THEN
|| 代表条件性的OR ELSE
如下图,参数1必须小于参数2,条件成立,&&则计算参数1到参数2之间所有数的和,||否则,提示参数1必须小于参数2
bash的测试类型
数值测试:
-gt: 是否大于;
-ge: 是否大于等于;
-eq: 是否等于;
-ne: 是否不等于;
-lt: 是否小于;
-le: 是否小于等于;
如下图,参数1必须-lt小于参数2,条件成立,才执行&&的条件,否则,提示参数1必须-lt小于参数2
字符串测试:
==:是否等于;
>: ascii码是否大于ascii码
<: 是否小于
!=: 是否不等于
=~: 左侧字符串是否能够被右侧的PATTERN所匹配
注意: 此表达式一般用于` `中;
-z "STRING":字符串是否为空,空为真,不空为假
-n "STRING":字符串是否不空,不空为真,空为假
注意:用于字符串比较时的用到的操作数都应该使用引号
文件存在性测试
-a FILE:同-e
-e FILE: 文件存在性测试,存在为真,否则为假;
存在性及类别测试
-b FILE:是否存在且为块设备文件;
-c FILE:是否存在且为字符设备文件;
-d FILE:是否存在且为目录文件;
-f FILE:是否存在且为普通文件;
-h FILE 或-L FILE:存在且为符号链接文件;
-p FILE:是否存在且为命名管道文件;
-S FILE:是否存在且为套接字文件;
文件权限测试:
-r FILE:是否存在且可读
-w FILE: 是否存在且可写
-x FILE: 是否存在且可执行
文件特殊权限测试:
-g FILE:是否存在且拥有sgid权限;
-u FILE:是否存在且拥有suid权限;
-k FILE:是否存在且拥有sticky权限;
文件大小测试:
-s FILE: 是否存在且非空;
文件是否打开:
-t fd: fd表示文件描述符是否已经打开且与某终端相关
-N FILE:文件自动上一次被读取之后是否被修改过(判断文件主要是看修改时间)
-O FILE:当前有效用户是否为文件属主
-G FILE:当前有效用户是否为文件属组
双目测试:
FILE1 -ef FILE2: FILE1与FILE2是否指向同一个设备上的相同inode
FILE1 -nt FILE2: FILE1是否新于FILE2;(判断文件主要是看修改时间)
FILE1 -ot FILE2: FILE1是否旧于FILE2;(判断文件主要是看修改时间)
组合测试条件
第一种方式:
COMMAND1 && COMMAND2 并且
COMMAND1 || COMMAND2 或者
! COMMAND 非
如:[ -e FILE ] && [ -r FILE ]
第二种方式:
EXPRESSION1 -a EXPRESSION2 并且
EXPRESSION1 -o EXPRESSION2 或者
! EXPRESSION