1 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
1.1 bash版本
查看bash的version
/bin/bash --version
1.2 sh与bash 的关系:
sh是一种POSIX标准,它有很多种实现,包括ksh88, dash,bash等。
因为sh是一种规范,并不是实现,所以/bin/sh实际上是一个硬链接,链接到某种实现上。大多数情况下,/bin/sh会链接到/bin/bash。所以执行 sh xx.sh 等价于执行 bash xx.sh
2 脚本书写规范
2.1 选择解释器
开头的"#!"字符又称为幻数,在执行bash脚本的时候,内核会根据"#!"后的解释器来确定该用那个程序解释这个脚本中的内容。
init.d目录。这个目录是干嘛的呢?
它归根结底只做了一件事情,但这件事情非同小可,是为整个系统做的,
因此它非常重要。init.d目录包含许多系统各种服务的启动和停止脚本。
head -1 /etc/init.d/*
init.d目录都是用来放服务脚本的,当Linux启动时,会寻找这些目录中的服务脚本,并根据脚本的run level确定不同的启动级别。
2.2 开发的规范
1) 放在统一的目录
2) 脚本以.sh为扩展名 (不是必需的)
3) 开头指定脚本解释器。(#! /bin/sh)
4) 开头加版本版权等信息,可配置~/.vimrc文件自动添加。
5) 脚本不要用中文注释,尽量用英文注释。
6) 代码书写优秀习惯
注意点:
a、成对的内容一次性写出来,防止遗漏,如[ ]、' '、" "等
b、[ ] 两端要有空格,先输入[ ],退格,输入2个空格,再退格写。
c、流程控制语句一次书写完,再添加内容。(if 条件 ; then 内容;fi)
d、通过缩进让代码易读。
f、脚本中的引号都是英文状态下的引号,其他字符也是英文状态。
语法结构为
if condtion then do something elif condtion then do something else do something fi
3 shell脚本的编写/执行/调试
3.1 创建第一个shell脚本文件
创建脚本文件 test.sh
vim test01.sh
在shell编程中,通常情况下,#代表注释,但是第一行的#是一个特例。
#! /bin/sh是shell脚本的一个标志,声明这个script使用的shell。
第一行的#!是一个约定标记, 它告诉脚本这段脚本需要什么解释器来执行.
第二行的echo命令则负责向屏幕上输出一句话。
运行这个脚本文件 sh test01.sh 或者是 ./test01.sh
刚开始执行不成功,是因为权限不够 如果赋权限 使用chmod命令
报错原因:
根据截图也可以看出,test01.sh的文字颜色是白色,并不是绿色,因此我们可以赋权限, 可以使用chmod 777 test01.sh命令来赋权限。
其实运行shell程序共有三种方法,除了给文件赋予可执行权限外,还有另外两种方法。
- chmod + x 使文件具有可执行权限, 直接运行;
- 直接调用解释器, 将脚本文件作为参数传入 (比如 bash test.sh )
- 使用source(也可用.代替)执行文件 (在当前bash环境下读取并执行FileName中的命令。该filename文件可以无"执行权限")
通常情况下, 最方便的方式就是方式1, 通过方式1执行你需要在脚本第一行写好这段脚本由哪个解释器来解释, 而通过方式2来执行则没有这个限制, 写了也没用。除此之外方式1与方式2执行命令就没有区别了
source script.sh:在脚本运行结束后,脚本中的变量在当前环境仍会被保留。
sh script.sh:在当前环境启动一个子进程运行脚本, 脚本中的变量会在脚本运行结束时释放掉。
文件权限解读
开头的-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
3.2 Shell的变量
3.2.1 变量的分类
变量可以分为三类:环境变量(全局变量)、普通变量(局部变量)、 特殊变量
环境变量:也可称为全局变量,可以在创建他们的Shell及其派生出来的任意子进程shell中使用,环境变量又可分为自定义环境变量和Bash**内置的环境变量
普通变量:也可称为局部变量,只能在创建他们的Shell函数或Shell脚本中使用。普通变量一般是由开发者用户开发脚本程序时创建的。
特殊变量:脚本内置的具有特殊用途的变量
使用 env ,export 命令查看系统中的环境变量
env 显示用户的环境变量
export 显示当前导出成用户变量的shell变量,并显示变量的属性(是否只读),
按变量名称排序
输出一个系统中的 环境变量 echo $HOME
3.2.2 普通变量
本地变量在用户当前的Shell生存期的脚本中使用。例如,本地变量a取值为1,这个值在用户当前Shell生存期中有意义。如果在Shell中启动另一个进程或退出,本地变量值将无效。
注意: $用来获取变量的值 ,=前后两端不能有空格
#! /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 "转义字符\"\""
注意:
单引号字符串的限制:单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的;
单引号字串中不能出现单独一个的单引号(对单引号使用转义符后也不行),但可成对出现,作为字符串拼接使用。
双引号里可以有变量,可以出现转义字符。
3.2.3 变量中引号的使用
只有在变量的值中有空格的时候,会使用引号。
单引号与双引号的区别在于,是否能够解析特殊符号。
变量使用反引号赋值,及使用${}获取参数值
取值的时候不加{}代表取的是某一个变量的值 :$ab 取的是ab变量的值
加{}之后取的是拼接之后的值:${a}b 取的是 ${a}的值拼接上字符串b
3.2.4 定义变量名技巧
- 变量名只能为字母、数字或下划线,只能以字母或下划线开头。
- 变量名的定义要有一定的规范,并且要见名知意。
示例:
ClsnAge=22 #<==每个单词的首字母大写的写法
clsn_age=22 #<==单词之间用"_"的写法
clsnAgeSex=man #<==驼峰语法:首个单词的首字母小写,其余单词首字母大写
CLSNAGE=22 #<==单词全大写的写法
- 一般的变量定义、赋值常用双引号;简单连续的字符串可以不加引号;希望原样输出时使用单引号。
- 希望变量的内容是命令的解析结果时,要用反引号``,或者用$()把命令括起来再赋值。
3.2.5 特殊变量
- 位置变量
常用的特殊位置参数说明
位置变量 |
作用说明 |
$0 |
获取当前执行的shell脚本的文件名,如果执行脚本带路径那么就包括脚本路径。 |
$n |
获取当前执行的shell脚本的第n个参数值,n=1..9,当n为0时表示脚本的文件名,如果n大于9用大括号括起来{10},参数以空格隔开。 |
$# |
获取当前执行的shell脚本后面接的参数的总个数 |
$* |
获取当前shell的所有传参的参数,不加引号同$@ (将接收到的每一个参数当做每一份数据,每个参数之间用空格来分开) 加上引号表示将传入的多个参数从整体上当做一份数据,以"1 2…n"的形式输出所有参数。 |
$@ |
获取当前shell的所有传参的参数,不加引号同$* 加上引号 表示仍然将传入的多个参数当做多份数据,空格区分,彼此之间独立,以"1" "2"…"n" 的形式输出所有参数。 |
当“$*”和“$@”都加双引号时,两者有区别,都不加双引号时,两者无区别。。 |
2. 进程状态变量
Shell进程的特殊状态变量说明
位置变量 |
作用说明 |
$? |
获取执行上一个指令的执行状态返回值(0为成功,非零为失败),这个变量最常用 |
$$ |
获取当前执行的Shell脚本的进程号(PID),这个变量不常用,了解即可 |
$! |
获取上一个在后台工作的进程的进程号(PID),这个变量不常用,了解即可 |
$_ |
获取在此之前执行的命令或脚本的最后一个参数,这个变量不常用,了解即可 |
1. echo**参数说明
参数 |
参数说明 |
-n |
不要追加换行 |
-e |
启用下列反斜杠转义的解释 |
-E |
显式地抑制对于反斜杠转义的解释 |
`echo' 对下列反斜杠字符进行转义: |
|
\n |
换行 |
\r |
回车 |
\t |
横向制表符 |
\b |
退格 |
\v |
纵向制表符 |
\c |
抑制更多的输出 参数 |
3.3定义变量的方式
3.3.1三种定义变量的方式
- 直接赋值
a=1
- 传参 (传递参数)
- 交互式设置变量,使用read命令 类似于Scanner
3.3.2 read命令说明
获取输入内容,在命令行中使用
3.4 变量的子串
3.4.1变量子串说明
表达式 |
说明 |
${parameter} |
返回变量$parameter的内容 |
${#parameter} |
返回变内容的长度(按字符),也适用于特殊变量 |
${parameter:offset} |
在变量${parameter}中,从位置offset之后开始提取子串到结尾 |
${parameter:offset:length} |
在变量${parameter}中,从位置offset之后开始提取长度为length的子串 |
${parameter#word} |
从变量${parameter}开头开始删除最短匹配的word子串 |
${parameterword} |
从变量${parameter}开头开始删除最长匹配的word子串 |
${parameter%word} |
从变量${parameter}结尾开始删除最短匹配的word子串 |
${parameter%%word} |
从变量${parameter}结尾开始删除最长匹配的word子串 |
${parameter/pattem/string} |
使用string代替第一个匹配的pattern |
${parameter//pattem/string} |
使用string代替所有匹配的pattern |
#!/bin/sh str="1abcdefghjabababWWWabbabab" echo "输出字符串:${str}" echo "字符的长度为:${#str}" echo "从第三位开始截取字符串:${str:3}" echo "从第三位开始截取6位:${str:3:6}" echo "从头开始删除最短匹配的ab:${str#ab}" echo "从头开始删除最长匹配的ab:${strab}" echo "从尾部开始删除最短匹配的ab:${str%ab}" echo "从尾部开始删除最长匹配的ab:${str%%ab}" echo "替换第一个ab:${str/ab/xxx}" echo "替换所有的ab:${str//ab/xxx}"
3.4.2 特殊扩展变量说明
表达式 |
说明 |
${parameter:-word} |
如果parameter的变量值为空或未赋值,则会返回word字符串并替代变量的值用途.如果变量未定义,则返回备用的值,防止变量为空值或因未定义而导致异常 |
${parameter:=word} |
如果parameter的变量值为空或未赋值,则设置这个变量值为word,并返回其值。位置变量和特殊变量不适用用途:基本同上一个${parameter>word},但该变量又额外给parameter变量赋值了 |
${parameter:?word} |
如果parameter变量值为空或未赋值,那么word字符串将被作为标准错误输出,否则输出变量的值。用途:用于捕捉由于变量未定义而导致的错误,并退出程序 |
${parameter:+word} |
如果parameter变量值为空或未赋值,则什么都不做,否则word字符串将替代变量的值 |
4 条件表达式
Dos2unix
4.1 &&、||
如果第一个命令执行成功(返回1),与 操作符 && 才会执行第二个命令
如果第一个命令执行不成功就直接返回0
如果第一个命令执行失败,或 操作符 || 才会执行第二个命令
如果第一个命令执行成功就直接返回1
4.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旧则为真,即测试表达式成立。根据文件的修改时间来计算 |
判断目录是否存在
7.3 字符串判断
字符串测试操作符
常用字符串测试操作符 |
说明 |
-n "字符串" |
若字符串的长度不为0,则为真,即测试表达式成立,n可以理解为no zero |
-z "字符串" |
若字符串的长度为0,则为真,即测试表达式成立,z可以理解为zero的缩写 |
"串 1"== "串 2" |
若字符串1等于字符串2,则为真,即测试表达式成立,可使用"=="代替"=" |
"串 1" != "串 2" |
若字符串1不等于字符串2,则为真,即测试表达式成立,但不能用"!=="代替"!=" |
1.对于字符串的测试,一定要将字符串加双引号之后再进行比较。 2.空格非空 |
#!/bin/bash # 判断字符串长度是否为0 [ -n "abc"] echo "[ -n 'abc' ]判断字符串长度是否为0:$?" [ -n ""] echo "[ -n '' ]判断字符串长度是否为0:$?" [ -n 0 ] echo "[ -n 0 ]判断字符串长度是否为0:$?" [ "a" == "a"] echo "[ \"a\"==\"a\" ]判断两个字符串是否相等:$?" [ "a" == "b"] echo "[ \"a\"==\"b\" ]判断两个字符串是否相等:$?"
练习:登录
#!/bin/bash # 登录 read -p "请输入用户名:" name read -p "请输入密码:" pwd [ "$name" == "admin" ] && [ "$pwd" == "123" ] && echo "登录成功!" || echo "登录失败"
使用
4.4 整数判断
整数二比较操作符参考
[[ == ]]
-eq []
在[]以及test中 使用的比较符号 |
在(())和[[]]中 使用的比较符号 |
说明 |
-eq |
==或= |
相等,全拼为equal |
-ne |
!= |
不相等,全拼为not equal |
-gt |
> |
大于,全拼为greater than |
-ge |
>= |
大于等于,全拼为greater equal |
-lt |
< |
小于,全拼为less than |
-le |
<= |
小于等于,全拼为less equal |
4.5 逻辑符号
&&:连接命令,如果第一个命令执行成功(返回1),与操作符&&才会执行第二个命令
&&:[[]]使用,作为逻辑运算符,判断两个表达式是否都成立
常用逻辑操作符
在[]和test中使用的操作符 |
说明 |
在[[]]和中使用的操作符 |
说明 |
-a |
[ 条件A -a 条件B ] A与B都要成立,整个表达式才成立 |
&& |
and,与,两端都为真,则结果为真 |
-o |
[ 条件A -o 条件B] A与B都不成立,整个表达式才不成立 |
|| |
or,或,两端有一个为真,则结果为真 |
! |
! |
not,非,两端相反,则结果为真 |
5 if条件语句
5.1单分支语句
if 条件表达式
then
Do something
fi
5.2双分支语句
if 条件表达式
then
。。。。。。
else
fi
5.3 多分支语句
if 条件
then
elif
then
else
fi
成绩:
#!/bin/bash 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
6 case条件结构语句
6.1 case语法结构
case "变量" in 值1) ;; 值2) ;; *) ;; esac
#!/bin/bash case $1 in 1) echo "第一季度" ;; 2) echo "第二季度" ;; *) echo "其他季度" esac
6.2 case与if的对比
case书写方式 case $name in 值1) 指令1 ;; 值2) 指令2 ;; *) 指令 esac if书写方式 if [ $name == "值1" ] then 指令1 elif [ $name == "值2" ] then 指令2 else 指令 fi
6.3 case值的书写方式
apple)
echo -e "$RED_COLOR apple $RES"
;;
也可以这样写,输入2种格式找同一个选项
apple|APPLE)
echo -e "$RED_COLOR apple $RES"
;;
7 for循环
7.1 列表for循环(常用)
#!/bin/bash for i in 取值列表 do 循环主体 命令done
7.2类似C语言的风格
for((exp1;exp2;exp3))
do
指令...
done
编写类似C语言风格脚本
Shell中的两种样式
求和:
11 while语法
基本结构:
声明一个变量
i=0; while 条件 do 命令 done
6 until循环
基本结构
until condition do do something... done
12 跳出循环
shell中也支持break跳出循环, continue跳出本次循环.用法与C, Java中相同
13 退出/返回状态
1) $?:返回上一条语句或脚本执行的状态
a. 0:成功
b. 1-255:不成功
2) exit 命令,用于退出脚本或当前Shell
exit 命令n 是一个从 0 到 255 的整数 ,0 表示成功退出,非零表示遇到某种失败 ,返回值 被保存在状态变量 $? 中 。
exit n
常见的返回状态码
a. 0: 执行正确
b. 1: 通用错误
c. 126: 命令或脚本没有执行权限
d. 127: 命令没找到
14 数据流重定向
标准输入:代码0,使用 < 或者 <<
标准输出:代码1,使用 > 或者 >>
错误输出:代码2,使用 > 或者 >>
> 表示 写入;>>表示追加
< 表示 读入;<<表示结束读入
EOF是END Of File的缩写,表示自定义终止符.
tail命令主要用于监视文件的内容
tail命令语法
tail [ -f ] [ -c Number | -n Number | -m Number | -b Number | -k Number ] [ File ]
参数解释:
-f 该参数用于监视File文件增长。
-c Number 从 Number 字节位置读取指定文件
-n Number 从 Number 行位置读取指定文件。
-m Number 从 Number 多字节字符位置读取指定文件,比方你的文件假设包括中文字,假设指定-c参数,可能导致截断,但使用-m则会避免该问题。
-b Number 从 Number 表示的512字节块位置读取指定文件。
-k Number 从 Number 表示的1KB块位置读取指定文件。
File 指定操作的目标文件名称
上述命令中,都涉及到number,假设不指定,默认显示10行。Number前面可使用正负号,表示该偏移从顶部还是从尾部開始计算。
tail可运行文件一般在/usr/bin/以下。
今天的文章 Shell 命令分享到此就结束了,感谢您的阅读。
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
如需转载请保留出处:https://bianchenghao.cn/bian-cheng-ji-chu/99854.html