什么是shell
Shell脚本语言是实现Linux/UNIX系统管理及自动化运维所必备的重要工具, Linux/UNIX系统的底层及基础应用软件的核心大都涉及Shell脚本的内容。Shell是一种编程语言, 它像其它编程语言如: C, Java, Python等一样也有变量/函数/运算符/if语句/循环控制/… 但在开始之前, 我想先理清Shell语言与Shell之间的关系.
当命令不在命令行中执行,而是从一个文件中执行时,该文件就是shell脚本。
Shell是一种解释型编程语言,不需要编译,执行时也是按行执行。
Shell脚本是由解释器解释执行的,常见的解释器有:bash dash ash ksh sh等
特点:
shell脚本是普通的文本文件,由流程控制逻辑和命令构成。
shell脚本通常以.sh作为后缀名,但不是必须的。
我们现阶段学习的主要是bash dash.
什么是shell脚本
命令、变量和流程控制语句等有机的结合起来
shell脚本擅长处理纯文本类型的数据,而linux中,几乎所有的配置文件,日志,都是纯文本类型文件。
清楚屏幕命令:clear(执行该指令后,终端上显示的内容将被清除,只留下一个空白屏幕);
脚本语言的种类(了解一下)
1、编译型语言
定义:指用专用的编译器,针对特定的操作平台(操作系统)将某种高级语言源代码一次性翻译成可被硬件平台直接运行的二进制机器码(具有操作数,指令、及相应的格式),这个过程叫做编译(./configure make makeinstall );编译好的可执行性文件(.exe),可在相对应的平台上运行(移植性差,但运行效率高)。。
典型的编译型语言有, C语言、C++等。
另外,Java语言是一门很特殊的语言,Java程序需要进行编译步骤,但并不会生成特定平台的二进制机器码,它编译后生成的是一种与平台无关的字节码文件(*.class)(移植性好的原因),这种字节码自然不能被平台直接执行,运行时需要由解释器解释成相应平台的二进制机器码文件;大多数人认为Java是一种编译型语言,但我们说Java即是编译型语言,也是解释型语言也并没有错。
2**、解释型语言**
定义:指用专门解释器对源程序逐行解释成特定平台的机器码并立即执行的语言;相当于把编译型语言的编译链接过程混到一起同时完成的。
解释型语言执行效率较低,且不能脱离解释器运行,但它的跨平台型比较容易,只需提供特定解释器即可。
常见的解释型语言有, Python(同时是脚本语言)与Ruby等。
3**、脚本语言**
定义:为了缩短传统的编写-编译-链接-运行(edit-compile-link-run)过程而创建的计算机编程语言。
特点:程序代码即是最终的执行文件,只是这个过程需要解释器的参与,所以说脚本语言与解释型语言有很大的联系。脚本语言通常是被解释执行的,而且程序是文本文件。
典型的脚本语言有,JavaScript,Python,shell等。
其他常用的脚本语句种类
shell脚本的优势在于处理操作系统底层的业务 (linux系统内部的应用都是shell脚本完成)因为有大量的linux系统命令为它做支撑。2000多个命令都是shell脚本编程的有力支撑,特别是grep、awk、sed等。例如:一键软件安装、优化、监控报警脚本,常规的业务应用,shell开发更简单快速,符合运维的简单、易用、高效原则.
shell解释器,用户和操作系统内核之间的桥梁
shell介于操作系统内核与用户之间,负责接收用户输入的操作指令(命令),并运行和解释,将需要执行的操作传递给操作系统内核并执行
shell程序在系统中充当了一个”命令解释“的角色
shell常见的种类(了解一下)
Bsh:由贝尔实验室编写。Bsh是产生较早的UNIX Shell程序,实现了最基本的命令解释器的功能,同时也可以作为脚本编程语言。
Csh:是因使用C语言的语法风格而得名,在用户的命令行交互界面上进行了很多改进,并增加了历史,别名,文件名替换,作业掏等功能,相比Bsh,Csh在更加适用为 用户提供命令交互操作。
Ksh:在Bsh和Csh之后出现的,结合了两都的功能优势,兼具Bsh的语法和Csh的交互特性。
Bash:从名称可以看出是Bsh的升级版本,是著名的开源软件项目,目前大多数的Linux版本(包括Red Hat公司的Linux系统)都使用Bash 作为默认的Shell程序当运行Shell程序时,实际运行的是Bash程序。
Zsh:更多地基于交互式操作进行设计的Shell程序,集成了Bash,Ksh等多种Shell程序的优点。
Linux默认shell是Bourne Again shell(bash)
cat /etc/shells 查看支持的shell的种类;
查看bash版本
/bin/bash --version
sh和bash的关系(了解一下)
sh是一种POSIX标准,它有很多种实现,包括ksh88, dash,bash等。
因为sh是一种规范,并不是实现,
所以/bin/sh实际上是一个硬链接,链接到某种实现上。大多数情况下,
/bin/sh会链接到/bin/bash。所以执行sh xx.sh
等价于执行
bash xx.sh
脚本书写规范
1.选择解释器
开头的"#!"字符又称为幻数,在执行bash脚本的时候,内核会根据"#!"后的解释器来确定该用那个程序解释这个脚本中的内容。
init.d目录。这个目录是干嘛的呢?
它归根结底只做了一件事情,但这件事情非同小可,是为整个系统做的,
因此它非常重要。init.d目录包含许多系统各种服务的启动和停止脚本。
init.d目录都是用来放服务脚本的,当Linux启动时,会寻找这些目录中的服务脚本,并根据脚本的run level确定不同的启动级别。
2.开发规范
1) 放在统一的目录;
2) 脚本以.sh为扩展名 (不是必需的);
3) 开头指定脚本解释器。(#! /bin/sh);
4) 开头加版本版权等信息,可配置~/.vimrc文件自动添加;
5) 脚本不要用中文注释,尽量用英文注释;
6) 代码书写优秀习惯。
注:
shell脚本规范:
1.脚本文件尽量放在同一个文件夹下面(方便好找);
2..sh为后缀;
3.成对的内容一次性写出来,防止遗漏,如[ ]、' '、" "等;
4.[ ] 两端要有空格,先输入[ ],退格,输入2个空格,再退格写;
5.流程控制语句一次书写完,再添加内容。(if 条件 ; then 内容;fi);
6.通过缩进让代码易读;
7.脚本中的引号都是英文状态下的引号,其他字符也是英文状态。
shell脚本编写 /执行 /测试
创建第一个shell脚本语言
第一步:先创建同一文件夹,mkdir shell
第二步:在创建以.sh为后缀的文件 touch test.sh
第三步:编写test.sh里边的内容, vim test.sh文件开头必须为 #!/bin/sh
echo命令是向屏幕输出内容
第四步:执行输出test.sh文件 sh test.sh 或者 ./test.sh
权限不够,文字颜色是白色,并不是绿色,我们可以使用chmod 777 文件名字命令来赋权限。
文件权限解读
开头的-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及其派生出来的任意子进程shell中使用,环境变量又可分为自定义环境变量和Bash**内置的环境变量**;(一般是在/etc/profile下定义的变量称为环境变量);
使用 env ,export 命令查看系统中的环境变量
env 显示用户的环境变量,export 显示当前导出成用户变量的shell变量,并显示变量的属性(是否只读);
按变量名称排序:输出一个系统中的 环境变量 echo $HOME;
(不用自己声明就可以使用的环境变量);
普通变量:也可称为局部变量,只能在创建他们的Shell函数或Shell脚本中使用。普通变量一般是由开发者用户开发脚本程序时创建的;(注:=左右两端一定不能有空格);
这个值在用户当前Shell生存期中有意义。如果在Shell中启动另一个进程或退出,本地变量值将无效。(切换用户是输出不出来值的,一旦切换shell生命就结束了);$是用来获取变量的值的
(开发人员自己声明的变量,变量名字=值);
特殊变量:脚本内置的具有特殊用途的变量;
(有特殊用途的变量叫做特殊变量 $0 $# $* );
1.位置变量
$0: 获取当前执行的shell脚本的文件名,如果执行脚本带路径那么就包括脚本路径;
$n: 获取当前执行的shell脚本的第n个参数值,n=1..9,当n为0时表示脚本的文件名,如果n大于9用大括号括起来{10},参数以空格隔开;(获取参数的值,可以直接写在后面)
$#: 获取当前执行的shell脚本后面接的参数的总个数;(脚本后面获取参数的个数,参数和参数使用空格隔开的)
$*: 获取当前shell的所有传参的参数,不加引号同$@(将接收到的每一个参数当做每一份数据,每个参数之间用空格来分开)加上引号表示将传入的多个参数从整体上当做一份数据,以"1 2…n"的形式输出所有参数;
不加双引号:显示为字符串;“ccbbddaattyyuuoo”
加双引号:显示单个字符;“cc” “bb” “dd” “aa” “tt” “yy” “uu” “oo”
$@: 获取当前shell的所有传参的参数,不加引号同$*,加上引号,表示仍然将传入的多个参数当做多份数据,空格区分,彼此之间独立,以"1" "2"…"n" 的形式输出所有参数。
在Shell中,∗和@都表示传递给脚本或函数的全部参数。二者的区别主要在于处理参数的方式上。
当∗和@不加双引号时,两者没有任何区别,都表示将接收到的每一个参数当做每一份数据,每个参数之间用空格来分开。
当∗和@加上双引号时,则两者就存在以下区别:
"∗"表示将传入的多个参数从整体上当做一份数据,以"1 2…n"的形式输出所有参数。
"@"表示仍然将传入的多个参数当做多份数据,空格区分,彼此之间独立,以"1" "2"…"n" 的形式输出所有参数。
$@和$*效果是一样的
(注:当“$*”和“$@”都加双引号时,两者有区别,都不加双引号时,两者无区别)
2.进程状态变量
$?:获取执行上一条指令的执行状态返回值(0为成功,非0为失败,异常)
$$:获取当前执行的Shell脚本的进程号(PID),这个变量不常用,了解即可;
$!:获取上一个在后台工作的进程的进程号(PID),这个变量不常用,了解即可;
$_:获取在此之前执行的命令或脚本的最后一个参数,这个变量不常用,了解即可;
echo 参数说明(了解一下即可)
三种定义变量的方式:
1.直接赋值: a=1
2.传参赋值(传递参数)
3.交互式设置变量,使用read命令 类似于Scanner
read命令说明:(相当于java中scanner在控制台手动输入,然后在获取下面是两种获取方法,那个方便用那个)
变量子串的说明:
单引号和双引号的区别
数字:单引号 、双引号、 或者不加、 结果是一样的 都是数字
字符串 :单引号和双引号 是一样的
取值: 单引号是原文输出 双引号是进行取值,不加也是进行取值;
定义转义字符:'\\' 第一个是转义字符符号,第二个才是要输出的内容;
注:
单引号字符串的限制:单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的;
单引号字串中不能出现单独一个的单引号(对单引号使用转义符后也不行),但可成对出现,作为字符串拼接使用。
双引号里可以有变量,可以出现转义字符。
变量中的引号使用
变量使用反引号赋值,及使用${}获取参数值(三种取值方法);
1.time=`date`; echo $time
2.time=`date`; echo ${time}
3.time=`date`; echo ${time}_b
条件表达式
&& :如果第一个命令执行成功返回1,然后执行第二个命令;
如果第一个命令执行失败返回0;
|| : 如果第一个命令执行成功返回1;
如果第一个命令执行失败返回0,然后在执行第二个命令;
文本判断
常用文件测试操作符:
判断目录是否存在 -d
判断文件是否存在 -f
判断是目录还是文件 -e
字符串判断测试
整数判断
整数二元比较操作
逻辑符号
常规逻辑符号操作
if条件语句
单分支语句
语法结构:
if 条件表达式
then
echo "输出语句"
fi
多分支语句
语法结构:
if 条件
then
elif
then
else
fi
case条件结构语句
语法结构:
case $1 in
1)
echo "xxx"
;;
2)
echo "xxx"
;;
3)
echo "xxx"
easac
(只能获取固定值)
shell脚本获取当前时间
for循环
for i in 取值列表
do
循环主体
done
思考:卸载JDK
第一步:查看是否安装JDK
rpm -qa|grep jdk
‘
第二步:找到JDK文件的路径
/usr/java
第三步:利用shell脚本卸载JDK
vim test.sh
最后一步运行test.sh脚本卸载JDK;
利用shell脚本实现1-100的和
./xx.sh
while语法
语法结构:
声明一个变量
i=0;
while(条件表达式)
do
命令
done
输出1-5
(while循环 :满足条件才执行)
(until循环 :满足条件不执行)
until循环
语法结构:
i=0
until(条件表达式):满足条件以后就不执行了
do
命令
done
跳出循环
shell中也支持break跳出循环, continue跳出本次循环.用法与C, Java中相同
break跳出当前循环;
continue跳出本次循环;
例子:break求出1-10的和大于20跳出循环,并打印;
continue,循环1-20,偶数跳过,奇数打印
退出 / 返回状态
1) $?:返回上一条语句或脚本执行的状态
a. 0:成功
b. 1-255:不成功
2) exit 命令,用于退出脚本或当前Shell
exit 命令n 是一个从 0 到 255 的整数 ,0 表示成功退出,非零表示遇到某种失败 ,返回值 被保存在状态变量 $? 中 。
exit n
常见的返回状态码
a. 0: 执行正确
b. 1: 通用错误
c. 126: 命令或脚本没有执行权限
d. 127: 命令没找到
数据流重定向
标准输入:代码0,使用 < 或者 <<
标准输出:代码1,使用 > 或者 >>
错误输出:代码2,使用 > 或者 >>
> 表示 写入; >>表示追加
<表示 读入;<<表示结束读入
EOF是END Of File的缩写,表示自定义终止符.