1.什么是shell
Shell是一种编程语言, 它像其它编程语言如: C, Java, Python等一样也有变量/函数/运算符/if语句/循环控制/… 但在开始之前, 我想先理清Shell语言与Shell之间的关系.
当命令不在命令行中执行,而是从一个文件中执行时,该文件就是shell脚本。
Shell是一种解释型编程语言,不需要编译,执行时也是按行执行。
Shell脚本是由解释器解释执行的,常见的解释器有:bash dash ash ksh sh等
特点:
shell脚本是普通的文本文件,由流程控制逻辑和命令构成。
shell脚本通常以.sh作为后缀名,但不是必须的。
我们现阶段学习的主要是bash dash.
2.什么是Shell脚本
命令、变量和流程控制语句等有机的结合起来
shell脚本擅长处理纯文本类型的数据,而linux中,几乎所有的配置文件,日志,都是纯文本类型文件
3. sh与bash 的关系:
sh是一种POSIX标准,它有很多种实现,包括ksh88, dash,bash等。
因为sh是一种规范,并不是实现,
所以/bin/sh实际上是一个硬链接,链接到某种实现上。大多数情况下,
/bin/sh会链接到/bin/bash。所以执行sh xx.sh
等价于执行
bash xx.sh
4. 开发的规范
1) 放在统一的目录
2) 脚本以.sh为扩展名 (不是必需的)
3) 开头指定脚本解释器。(#! /bin/sh)
4) 开头加版本版权等信息,可配置~/.vimrc文件自动添加。
5) 脚本不要用中文注释,尽量用英文注释。
6) 代码书写优秀习惯
a、成对的内容一次性写出来,防止遗漏,如[ ]、' '、" "等
b、[ ] 两端要有空格,先输入[ ],退格,输入2个空格,再退格写。
c、流程控制语句一次书写完,再添加内容。(if 条件 ; then 内容;fi)
语法结构为
if condtion
then
do something
elif condtion
then
do something
else
do something
fi
d、通过缩进让代码易读。
f、脚本中的引号都是英文状态下的引号,其他字符也是英文状态。
g、定义变量时,=两端不能有空格
5. 创建第一个shell脚本文件
创建脚本文件test.sh
[root@localhost ~]# mkdir test
[root@localhost ~]# cd test
[root@localhost test]# ls
[root@localhost test]# touch test.sh
在shell编程中,通常情况下,#代表注释,但是第一行的#是一个特例。
#! /bin/sh是shell脚本的一个标志,声明这个script使用的shell。(script通常指的是一段可执行的程序代码)
第一行的#!是一个约定标记, 它告诉脚本这段脚本需要什么解释器来执行.
第二行的echo命令则负责输出语句。
运行这个脚本文件 sh test.sh 或者是 ./test.sh
刚开始执行不成功,是因为权限不够 如果赋权限 使用chmod命令
报错原因:
根据截图也可以看出,text01.sh的文字颜色是白色,并不是绿色,因此我们可以赋权限, 可以使用chmod 777 test.sh命令来赋权限。
其实运行shell程序共有三种方法,除了给文件赋予可执行权限外,还有另外两种方法。
- chmod +xxx,使文件具有可执行权限, 直接运行;
- 直接调用解释器, 将脚本文件作为参数传入 (比如bash test.sh)
- 使用source(也可用.代替)执行文件 (在当前bash环境下读取并执行FileName中的命令。该filename文件可以无"执行权限")
通常情况下, 最方便的方式就是方式1, 通过方式1执行你需要在脚本第一行写好这段脚本由哪个解释器来解释, 而通过方式2来执行则没有这个限制, 写了也没用。除此之外方式1与方式2执行命令就没有区别了,
source script.sh:在脚本运行结束后,脚本中的变量在当前环境仍会被保留。
sh script.sh:在当前环境启动一个子进程运行脚本, 脚本中的变量会在脚本运行结束时释放掉。
6.文件权限解读
开头的-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. Shell的变量
变量可以分为三类:环境变量(全局变量)、普通变量(局部变量)、 特殊变量
环境变量:也可称为全局变量,可以在创建他们的Shell及其派生出来的任意子进程shell中使用,环境变量又可分为自定义环境变量和Bash**内置的环境变量**
普通变量:也可称为局部变量,只能在创建他们的Shell函数或Shell脚本中使用。普通变量一般是由开发者用户开发脚本程序时创建的。
特殊变量:脚本内置的具有特殊用途的变量
使用 env ,export 命令查看系统中的环境变量
env 显示用户的环境变量
export 显示当前导出成用户变量的shell变量,并显示变量的属性(是否只读),
按变量名称排序
输出一个系统中的 环境变量 echo $HOME
7.1 普通变量
本地变量在用户当前的Shell生存期的脚本中使用。例如,本地变量a取值为1,这个值在用户当前Shell生存期中有意义。如果在Shell中启动另一个进程或退出,本地变量值将无效。
例:a=1
注意:$用来获取变量的值 ,=前后两端不能有空格
7.2 定义变量名技巧
- 变量名只能为字母、数字或下划线,只能以字母或下划线开头。
- 变量名的定义要有一定的规范,并且要见名知意。
示例:
ClsnAge=22 #clsn_age=22 #clsnAgeSex=man #CLSNAGE=22 #
- 一般的变量定义、赋值常用双引号;简单连续的字符串可以不加引号;希望原样输出时使用单引号。
- 希望变量的内容是命令的解析结果时,要用反引号``,或者用$()把命令括起来再赋值。
7.3 特殊变量
- 位置变量
常用的特殊位置参数说明
位置变量 | 作用说明 |
$0 | 获取当前执行的shell脚本的文件名,如果执行脚本带路径那么就包括脚本路径。 |
$n | 获取当前执行的shell脚本的第n个参数值,n=1..9,当n为0时表示脚本的文件名,如果n大于9用大括号括起来{10},参数以空格隔开。 |
$# | 获取当前执行的shell脚本后面接的参数的总个数 |
$* | 获取当前shell的所有传参的参数,不加引号同$@ (将接收到的每一个参数当做每一份数据,每个参数之间用空格来分开) 加上引号表示将传入的多个参数从整体上当做一份数据,以"1 2…n"的形式输出所有参数。 |
$@ | 获取当前shell的所有传参的参数,不加引号同$* 加上引号 表示仍然将传入的多个参数当做多份数据,空格区分,彼此之间独立,以"1" "2"…"n" 的形式输出所有参数。 |
当“$*”和“$@”都加双引号时,两者有区别,都不加双引号时,两者无区别。。 |
8.单引号和双引号:
数字 单引号 双引号 或者是不加 结果是一样的 都是数字
单引号定义字符串所见即所得,即将单引号内的内容原样输出
双引号引用的内容,所见非所得。如果内容中有命令、变量等,会先把变量、命令解析出结果,然后在输出最终内容。
字符串,常量 单引号和双引号是一样的
字符串常量使用单引号括起来,如果字符串中含有变量、命令等使用双引号括起来,不建议不加引号。
单引号字串中不能出现单独一个的单引号(对单引号使用转义符后也不行),但可成对出现,作为字符串拼接使用。
双引号里可以有变量,可以出现转义字符。
8.1 变量中引号的使用
只有在变量的值中有空格的时候,会使用引号。
单引号与双引号的区别在于,是否能够解析特殊符号。
变量使用反引号赋值,及使用${}获取参数值
取值的时候不加{}代表取的是某一个变量的值 :$ab 取的是ab变量的值
加{}之后取的是拼接之后的值:${a}b 取的是 ${a}的值拼接上字符串b
赋值:
1. 直接赋值 a=1
2. 传参赋值 a=$1
3. 交互赋值 read
取值: $REPLY
Read str
取值:$str
- 第一个命令没有指定变量名,因此用户输入的内容会被赋值给默认的变量REPLY。用户输入的内容将存储在变量REPLY中。
- 第二个命令指定了变量名为str,因此用户输入的内容会被赋值给变量str。用户输入的内容将存储在变量str中。
- 在代码的后续部分,可以使用$REPLY来获取第一个read命令中用户输入的值,使用$str来获取第二个read命令中用户输入的值。
&& ||
&& 当第一个条件成立的时候 1 再 执行第二个条件
第一个条件不成立 直接返回0
|| 当第一个条件不成立的时候 才会执行第二个条件
当第一个条件成立的时候 就不执行第二个条件
9.if条件语句
9.1单分支语句
if 条件表达式
then
Do something
fi
9.2双分支语句
if 条件表达式
then
。。。。。。
else
fi
9.3多分支语句
if 条件
then
elif
then
else
fi