一、什么是shell
Shell是一个由C语言编写的程序,它相当于一个IDE开发环境,这个环境是用C语言编程的,它用的是脚本语言,一些Linux命令集合。
它的主要作用就是在用户和操作系统之间搭起一道桥梁,shell介于操作系统内核与用户之间,负责接收用户输入的操作指令(命令),并运行和解释,将需要执行的操作传递给操作系统内核并执行。
二、什么是Shell脚本
Shell脚本是将命令、变量和流程控制语句等有机的结合起来。
Shell脚本擅长处理纯文本类型的数据,而Linux中,几乎所有的配置文件,日志,都是纯文本类型文件。
Shell脚本语言是实现Linux/UNIX系统管理及自动化运维所必须的重要工具,Linux/UNIX系统的底层及基础应用软件的核心大都涉及Shell脚本的内容。
Shell脚本是由解释器解释执行的,常见的解释器有:bash dash ash ksh sh等,我们现阶段学习的主要是bash dash。
特点:
Shell脚本是普通的文本文件,由流程控制逻辑和命令构成。
Shell脚本通常以.sh作为后缀名,但不是必须的。
优势:
shell脚本的优势在于处理操作系统底层的业务 (linux系统内部的应用都是shell脚本完成)因为有大量的linux系统命令为它做支撑。2000多个命令都是shell脚本编程的有力支撑,特别是grep、awk、sed等。例如:一键软件安装、优化、监控报警脚本,常规的业务应用,shell开发更简单快速,符合运维的简单、易用、高效原则.
三、脚本语言
定义:脚本语言是为了缩短传统的编写-编译-链接-运行(edit-compile-link-run)过程而创建的计算机编程语言。
特点:程序代码即是最终的执行文件,只是这个过程需要解释器的参与,所以说脚本语言与解释型语言有很大的联系。脚本语言通常是被解释执行的,而且程序是文本文件。
典型的脚本语言有,JavaScript,Python,shell等。
四、Shell常见种类
在 Linux 中,常见的 Shell 种类如下:
-
Bash (Bourne Again SHell):Bash 是 Bourne Shell 的增强版,是大多数 Linux 系统的默认 Shell。它是一种功能强大且广泛使用的 Shell,提供了命令历史记录、自动补全、括号扩展等特性。
-
Zsh (Z Shell):Zsh 是一种功能强大的 Shell,提供了比 Bash 更多的高级功能,例如更强大的命令补全、自动纠错、主题定制等。Zsh 在配置和扩展方面非常灵活,并且具有良好的用户体验。
-
Ksh (Korn Shell):Ksh 是一种功能丰富的 Shell,由 AT&T Bell Labs 的 David Korn 开发。它是 Bash 的前身之一,提供了更多的编程特性和高级功能。
-
Csh (C Shell):Csh 是一种基于 C 语言语法的 Shell,具有与 C 语言相似的语法结构。它提供了别名定义、历史命令替换和脚本编写能力,适用于特定的用例和个人偏好。
-
Tcsh (TENEX C Shell):Tcsh 是 Csh 的一种改进版本,提供了一些额外的特性,如命令补全和编辑模式切换。
Linux默认shell是Bourne Again shell(bash)
命令:cat /etc/shells 列出可用的 shell 的路径
命令:/bin/bash --version 查看bash的版本
五、sh与bash的关系
sh 和 bash 是两种不同的 Shell,但是它们之间有一些关系。
Bash(Bourne-Again SHell)是 sh 的增强版,也就是说 bash 具有 sh 的所有功能,并且增加了更多的功能和特性。Bash 支持递归函数、数组、命令行编辑等功能,同时还提供了很多内置命令和扩展工具。Bash 在使用上与 sh 基本相同,只是在一些语法和功能上有所区别。
因为sh是一种规范,并不是实现,所以/bin/sh实际上是一个符号链接,指向 /bin/bash 或者其它类似的 shell,大多数情况下,/bin/sh会链接到/bin/bash,所以执行sh xx.sh等价于执行 bash xx.sh。
六、脚本书写规范
1.放在同一的目录里
2.脚本以.sh为扩展名 (不是必需的)
3.开头指定脚本解释器。(#! /bin/sh)
4.开头加版本版权等信息,可配置~/.vimrc文件自动添加。
5.脚本不要用中文注释,尽量用英文注释。
6.代码书写优秀习惯
a、成对的内容一次性写出来,防止遗漏,如[ ]、' '、" "等
b、[ ] 两端要有空格
c、流程控制语句一次书写完,再添加内容。(if 条件 ; then 内容;fi)
d、通过缩进让代码易读。
f、脚本中的引号都是英文状态下的引号,其他字符也是英文状态。
七、shell脚本的编写/执行/调试
1.创建shell脚本文件
命令:touch 文件名.sh
范例:创建一个名为b的shell脚本文件
touch b.sh
2.shell脚本文件的编写
在shell编程中,通常情况下,#代表注释,但是第一行的#是一个特例。
#! /bin/sh是shell脚本的一个标志,声明这个script使用的shell。
第一行的#!是一个约定标记, 它告诉脚本这段脚本需要什么解释器来执行.
第二行的echo命令则负责向屏幕上输出一句话。
3.shell脚本文件的运行
命令: sh test.sh 或者是 ./test.sh
sh test.sh:在当前环境启动一个子进程运行脚本, 脚本中的变量会在脚本运行结束时释放掉。
刚开始 ./test.sh 执行不成功,是因为权限不够 如果赋权限 使用chmod命令来赋权限
其实运行shell程序共有三种方法,除了给文件赋予可执行权限外,还有另外两种方法。
1.chmod +x使文件具有可执行权限, 直接运行;
2.直接调用解释器, 将脚本文件作为参数传入 (比如bash test.sh)
3.使用source(也可用.代替)执行文件 (在当前bash环境下读取并执行FileName中的命令。该filename文件可以无"执行权限")
source test.sh:在脚本运行结束后,脚本中的变量在当前环境仍会被保留。
通常情况下, 最方便的方式就是方式1, 通过方式1执行你需要在脚本第一行写好这段脚本由哪个解释器来解释, 而通过方式2来执行则没有这个限制, 写了也没用。除此之外方式1与方式2执行命令就没有区别了。
八、文件权限解读
文件开头的-rwxrw-r--这一字符串标识文件权限。
这个字符串有10位,可以分为4段来解读。注:r--可读,w--可写,x--可执行。
第一段(第1位)表示是目录还是文件,-表示是文件,d表示是目录;
第二段(第2-4位,共3个字符串)表示文件所属用户对它的权限;
第三段(第5-7位,共3个字符串)表示文件所属用户组用户对它的权限;
第四段(第8-10位,共3个字符串)表示其他用户对它的权限;
读取权限 r = 4 | 写入权限 w = 2 | 执行权限 x = 1 |
775 这三个数字代表拥有者,组用户,其他用户的权限。
例如:
7 拥有者有 读取,写入,执行权限
7 组用户有 读取,写入,执行权限
5 其他用户有 读取,执行权限(4+1 = 5)
777 与 775的区别是
其他用户有写入权限,而775的没有。
举个例子:
文件A,权限是775
root是拥有者
www-data是组用户
ooo 是其他用户
那么
root,www-data,ooo都能对文件A读取和执行。
root,www-data 能对文件A写入。
ooo不能对文件A写入。
如果是777
则三个用户都能读取,写入,执行文件A
九、Shell的变量
变量可以分为三类:环境变量(全局变量)、普通变量(局部变量)、 特殊变量
环境变量:也可称为全局变量,可以在创建他们的Shell及其派生出来的任意子进程shell中使用,环境变量又可分为自定义环境变量和Bash**内置的环境变量**
普通变量:也可称为局部变量,只能在创建他们的Shell函数或Shell脚本中使用。普通变量一般是由开发者用户开发脚本程序时创建的。
特殊变量:脚本内置的具有特殊用途的变量
使用 env ,export 命令查看系统中的环境变量
env 显示用户的环境变量
export 显示当前导出成用户变量的shell变量,并显示变量的属性(是否只读),按变量名称排序
输出一个系统中的 环境变量 echo $HOME
1.普通变量
本地变量在用户当前的Shell生存期的脚本中使用。例如,本地变量a取值为1,这个值在用户当前Shell生存期中有意义。如果在Shell中启动另一个进程或退出,本地变量值将无效。
2.变量中引号的使用
单引号字符串的限制:单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的;
单引号字串中不能出现单独一个的单引号(对单引号使用转义符后也不行),但可成对出现,作为字符串拼接使用。
双引号里可以有变量,可以出现转义字符。
#! /bin/sh
#输出数字,字符串
echo "不带引号的数字:"
echo 1
echo "带单引号的数字:"
echo '1'
echo '双引号数字:'
echo "1"
echo "带单引号字符串:"
echo 'hello'
echo "双引号字符串:"
echo "hello"
#定义变量
str="123"
#输出变量
echo $str
echo "单引号变量:"
echo '$str'
echo '双引号变量:'
echo "$str"
#定义转义字符
echo '转义字符\'\'
echo "转义字符\"\""
注意:$用来获取变量的值 ,=前后两端不能有空格
变量可以使用反引号赋值,及使用${}获取参数值
取值的时候不加{}代表取的是某一个变量的值 :$ab 取的是ab变量的值,加{}之后取的是拼接之后的值:${a}_b 取的是 ${a}的值拼接上字符串_b
3.定义变量名技巧
1.变量名只能为字母、数字或下划线,只能以字母或下划线开头。
2.变量名的定义要有一定的规范,并且要见名知意。
示例:
ClsnAge=22 #<==每个单词的首字母大写的写法
clsn_age=22 #<==单词之间用"_"的写法
clsnAgeSex=man #<==驼峰语法:首个单词的首字母小写,其余单词首字母大写
CLSNAGE=22 #<==单词全大写的写法
1.一般的变量定义、赋值常用双引号;简单连续的字符串可以不加引号;希望原样输出时使用单引号。
2.希望变量的内容是命令的解析结果时,要用反引号'',或者用$()把命令括起来再赋值。
4.特殊变量
1.位置变量
常用的特殊位置参数说明
位置变量 | 作用说明 |
$0 | 获取当前执行的shell脚本的文件名,如果执行脚本带路径那么就包括脚本路径。 |
$n | 获取当前执行的shell脚本的第n个参数值,n=1..9,当n为0时表示脚本的文件名,如果n大于9用大括号括起来{10},参数以空格隔开。 |
$# | 获取当前执行的shell脚本后面接的参数的总个数 |
$* | 获取当前shell的所有传参的参数,不加引号同如果给加上双引号,例如: “$”,则表示将所有的参数视为单个字符串,相当于“112$3”。 12 34 56 |
$@ | 获取当前shell的所有传参的参数,不加引号同如果给@加上双引号,例如: “则表示将所有参数视为不同的独立字符串,相当于1” “3” “……”,这是将参数传递给其他程序的最佳方式,因为他会保留所有内嵌在每个参数里的任何空白。 12 34 56 |
当“$*”和“$@”都加双引号时,两者有区别,都不加双引号时,两者无区别。。 |
2. 进程状态变量
Shell进程的特殊状态变量说明
位置变量 | 作用说明 |
$? | 获取执行上一个指令的执行状态返回值(0为成功,非零为失败),这个变量最常用 |
$$ | 获取当前执行的Shell脚本的进程号(PID),这个变量不常用,了解即可 |
$! | 获取上一个在后台工作的进程的进程号(PID),这个变量不常用,了解即可 |
$_ | 获取在此之前执行的命令或脚本的最后一个参数,这个变量不常用,了解即可 |
echo**参数说明**
参数 | 参数说明 |
-n | 不要追加换行 |
-e | 启用下列反斜杠转义的解释 |
-E | 显式地抑制对于反斜杠转义的解释 |
`echo' 对下列反斜杠字符进行转义: | |
\n | 换行 |
\r | 回车 |
\t | 横向制表符 |
\b | 退格 |
\v | 纵向制表符 |
\c | 抑制更多的输出 参数 |
5.定义变量的方式
三种定义变量的方式
1.直接赋值 a=1
2.传参 (传递参数)
3.交互式设置变量,使用read命令 类似于Scanner
6.read命令说明
获取输入内容,在命令行中使用
7.变量的子串
变量子串说明
表达式 | 说明 |
${parameter} | 返回变量$parameter的内容 |
${#parameter} | 返回变内容的长度(按字符),也适用于特殊变量 |
${parameter:offset} | 在变量${parameter}中,从位置offset之后开始提取子串到结尾 |
${parameter:offset:length} | 在变量${parameter}中,从位置offset之后开始提取长度为length的子串 |
${parameter#word} | 从变量${parameter}开头开始删除最短匹配的word子串 |
${parameter##word} | 从变量${parameter}开头开始删除最长匹配的word子串 |
${parameter%word} | 从变量${parameter}结尾开始删除最短匹配的word子串 |
${parameter%%word} | 从变量${parameter}结尾开始删除最长匹配的word子串 |
${parameter/pattem/string} | 使用string代替第一个匹配的pattern |
${parameter//pattem/string} | 使用string代替所有匹配的pattern |
#!/bin/sh
str="1bcabdbdbdffgjrjhrhgotababab"
echo "输出字符串:${str}"
echo "字符的长度为:${#str}"
echo "从第三位开始截取字符串:${str:3}"
echo "从第三位开始截取6位:${str:3:6}"
echo "从头开始删除最短匹配的ab:${str#ab}"
echo "从头开始删除最长匹配的ab:${str##ab}"
echo "从尾部开始删除最短匹配的ab:${str%ab}"
echo "从尾部开始删除最长匹配的ab:${str%%ab}"
echo "替换第一个ab:${str/ab/xxx}"
echo "替换所有的ab:${str//ab/xxx}"
8.特殊扩展变量说明
表达式 | 说明 |
${parameter:-word} | 如果parameter的变量值为空或未赋值,则会返回word字符串并替代变量的值用途.如果变量未定义,则返回备用的值,防止变量为空值或因未定义而导致异常 |
${parameter:=word} | 如果parameter的变量值为空或未赋值,则设置这个变量值为word,并返回其值。位置变量和特殊变量不适用用途:基本同上一个${parameter>word},但该变量又额外给parameter变量赋值了 |
${parameter:?word} | 如果parameter变量值为空或未赋值,那么word字符串将被作为标准错误输出,否则输出变量的值。用途:用于捕捉由于变量未定义而导致的错误,并退出程序 |
${parameter:+word} | 如果parameter变量值为空或未赋值,则什么都不做,否则word字符串将替代变量的值 |
十、条件表达式
1.&&,||
如果第一个命令执行成功(返回1),与操作符&&才会执行第二个命令
如果第一个命令执行不成功就直接返回0
如果第一个命令执行失败,或操作符||才会执行第二个命令
如果第一个命令执行成功就直接返回1
2.文件判断
常用文件测试操作符
常用文件测试操作符 | 说明 |
-d文件,d的全拼为directory | 文件存在且为目录则为真,即测试表达式成立 |
-f文件,f的全拼为file | 文件存在且为普通文件则为真,即测试表达式成立 |
-e文件,e的全拼为exist | 文件存在则为真,即测试表达式成立。注意区别于“-f”,-e不辨别是目录还是文件 |
-r文件,r的全拼为read | 文件存在且可读则为真,即测试表达式成立 |
-s文件,s的全拼为size | 文件存在且文件大小不为0则为真,即测试表达式成立 |
-w文件,w的全拼为write | 文件存在且可写则为真,即测试表达式成立 |
-x文件,x的全拼为executable | 文件存在且可执行则为真,即测试表达式成立 |
-L文件,L的全拼为link | 文件存在且为链接文件则为真,即测试表达式成立 |
fl -nt f2,nt 的全拼为 newer than | 文件fl比文件f2新则为真,即测试表达式成立。根据文件的修改时间来计算 |
fl -ot f2,ot 的全拼为 older than | 文件fl比文件f2旧则为真,即测试表达式成立。根据文件的修改时间来计算 |
3.字符串判断
字符串测试操作符
常用字符串测试操作符 | 说明 |
-n "字符串" | 若字符串的长度不为0,则为真,即测试表达式成立,n可以理解为no zero |
-z "字符串" | 若字符串的长度为0,则为真,即测试表达式成立,z可以理解为zero的缩写 |
"串 1"== "串 2" | 若字符串1等于字符串2,则为真,即测试表达式成立,可使用"=="代替"=" |
"串 1" != "串 2" | 若字符串1不等于字符串2,则为真,即测试表达式成立,但不能用"!=="代替"!=" |
1.对于字符串的测试,一定要将字符串加双引号之后再进行比较。 2.空格非空 |
4.整数判断
整数二元比较操作符参考
[[ == ]]
-eq []
在[]以及test中 使用的比较符号 | 在(())和[[]]中 使用的比较符号 | 说明 |
-eq | ==或= | 相等,全拼为equal |
-ne | != | 不相等,全拼为not equal |
-gt | > | 大于,全拼为greater than |
-ge | >= | 大于等于,全拼为greater equal |
-lt | < | 小于,全拼为less than |
-le | <= | 小于等于,全拼为less equal |
5.逻辑符号
&&:连接命令,如果第一个命令执行成功(返回1),与操作符&&才会执行第二个命令
常用逻辑操作符
在[]和test中使用的操作符 | 说明 | 在[[]]和中使用的操作符 | 说明 |
-a | [ 条件A -a 条件B ] A与B都要成立,整个表达式才成立 | && | and,与,两端都为真,则结果为真 |
-o | [ 条件A -o 条件B] A与B都不成立,整个表达式才不成立 | || | or,或,两端有一个为真,则结果为真 |
! | ! | not,非,两端相反,则结果为真 |
十一、if条件语句
1.单分支语句
if 条件表达式
then
Do something
fi
2.双分支语句
if 条件表达式
then
。。。。。。
else
fi
3多分支语句
if 条件
then
elif
then
else
fi
#>=90优秀
#>=80 良好
#>=70 中等
#>=60 及格
#! /bin/sh
if [ $# == 0 ]
then
echo "没有参数"
else
if [ $1 -ge 90 ]
then
echo "非常优秀!!!"
elif [ $1 -ge 80 ]
then
echo "优秀"
elif [ $1 -ge 70 ]
then
echo "良好"
elif [ $1 -ge 60 ]
then
echo "及格"
else
echo "不及格"
fi
fi
十二、for循环
1.列表for循环(常用)
for i in 取值列表
do
循环主体
命令done
案例1:循环取出 123
案例2:/tmp 目录下面找到含有test 文件并输出出来