# Shell脚本概述 将要执行的命令吧顺序保存到一个文本文档 给该文件可执行权限,便可运行 可结合各种Shell控制语句以完成更复杂的操作
Shell脚本应用场景:
重复性操作
批量事物处理
自动化运维
服务运行状态监控(zabbix)
定时任务执行
Shell脚本的作用:
命令解释器,翻译官
介于系统内核与用户之间,负责解释命令行
shell 控制Linux内核,内核态会加载系统硬件提供支持,然后输出信息 (结果、bug、err)
shell 也可以通过控制应用程序,再进行控制Linux内核
应用程序(例如kickstart 无人值守)
内核放在boot中,boot还存放的GRUB菜单
用户的登录Shell
登录后默认使用的Shell程序,一般为/bin/bash
不同的Shell的内部指令,运行环境等会有所区别
[root@localhost ~]# cat /etc/shells
/bin/sh
/bin/bash
/usr/bin/sh
/usr/bin/bash
/bin/tcsh
/bin/csh
编写脚本代码
使用vi文本编辑器
每行一条Linux命令,按执行顺序依次编写
shell脚本的执行通常有以下几种方式:
方法一:当前路径(决定路径与相对路径)下执行脚本(要有执行权限)/home/.sh或者./ first.sh
方法二: sh . bash脚本文件路劲(这种方式可以不对脚本文件添加执行权限)bash first .sh或sh first .sh
方法三: source脚本文件路劲(可以没有执行权限)source first.sh
方法四:其他方法
sh < first .sh或者cat first.sh lsh ( bash)
重定向:
用户通过操作系统处理信息的过程中,包括以下几类交互设备文件标准输入(STDIN):默认的设备是键盘,文件编号为0,命令将从标准输入文件中读取在执行过程中需要的输入数据。
标准输出(STDOUT):默认的设备是显示器,文件编号为1,命令将执行后的输出结果发送到标准输出文件。
标准错误(STDERR):默认的设备是显示器,文件编号为2,命令将执行期间的各种错误信息发送到标准错误文件。
从键盘接收用户输入的各种命令字串、辅助控制信息,并将命令结果输出到屏幕上;如果命令执行出错,也会将错误信息反馈到屏幕上
重定向输出
重定向输出指的是将命令的正常输出结果保存到指定的文件中,而不是直接显示在显示器的屏幕上.。
重定向输出使用">“或”>>"操作符号,分别用于覆盖或追加文件
若重定向输出的目标文件不存在,则会新建该文件,然后将前面命令的输出结果保存到该文件巾:若目标文件已经存在,则将输出结果覆盖或追加到文件中。
意思是当原来文件中有内容的话,原来的内容公被覆盖掉
意思是当原来文件中有内容的话,新加的内容会追加到里面而不会覆盖原来的内容
例1
uname-p > kernel.txt
#查看结果
cat kernel.txt
重定向输入
重定向输入指的是将命令中接收输入的途径由默认的键盘改为指定的文件,而不是等待从键磁输入。重定向输入使用“<操作符。
通过重定向输入可以使一些交互式操作过程能够通过读取文件来完成。例如,使用passwd命令为用户设置密码时,每次都必须根据提示输入两次密码字串,非常烦琐,若改
用重定向输入将可以省略交互式的过程,而自动完成密码设置(结合passwd
命令的"–stdin”选项来识别标准输入)。
例1
#添加初始密码串内容"123456"123456vim pass.txt
#从pass.txt读入密码给paswd命令而不是等待用户从键盘输入123456passwd --stdin jerry < pass.txt
例2:
[root@localhost ~]# cat 1.txt//以键盘为输入设备,这也是系统默认的1234
1234
[ root@localhost ~] # cat <1.txt//跟cat 1.txt结果是一样的,但是这是以1.txt文件作为输入设备了1234
1234
默认情况下. cat
命令会接受标准输入设备(键盘)的输入,并显示到控制台,但如果用文件代替键盘作为输入设备,那么该命令会以指定的文件作为输入设备,并将文件中的内容读取并显示到控制台
[ root@localhost ~]#cat <<0
//以0作为分界符,只要不输入0就会一直输入数据从而显示到屏幕
123456
0
[ root@localhost ~] #cat << 0 >a.txt
//可以把输入重定向和输出重定向结合使用,把从屏幕输出的内容保存到文件
123
456
0
[ root@localhost ~]# cat a.txt
123
456
错误重定向
错误重定向指的是将执行命令过程中出现的错误信息(如选项或参数错误等)保存到指定的文件,而不是直接显示在屏幕上。错误重定向使用"2>”操作符
2个作用:
在实际应用中,错误重定向可用来收集程序执行的错误信息,为排错提供依据。
还可以将无关紧要的错误信息重定向到空文件/dev/null 中,以保持脚本输出的简洁。
使用"2>“操作符时,会像使用”>“操作符一样覆盖目标文件的内容,若要追加内容而不是覆盖文件,则应改用”2>>”操作有
当命令输出的结果可能既包括标准输出(正常执行)信息,又包括错误输出信息时,可
以使用操作符”>“"2>“将两类输出信息分别保存到不同的文件,也可以使用”&>"操作符将两类输出信息保存到同一个文件。
/dev/null:把它看作"黑洞",所有写入它的内容都会永远丢失.
而尝试从它那儿读取内容则什么也读不到.然而/dev/null对命令行和脚本都非常的有用
echo kgc > /dev/ null
cat l dev / null #什么信息也看不到
&>和>&符号
&表示等同于的意思
把正确和错误的消息输入到相同的位置1>&2把标准输出重定向到标准错误
2>&1把标准错误重定向到标准输出,
例1:将错误显示的内容和正确显示的内容分开ls /etc/passwd xxx
ls:无法访问xxx:没有那个文件或目录
/etc/passwd
ls /etc/passwd > a.txt
ls:无法访问xxx:没有那个文件或目录cat a .txt
/ etc/ passwd
ls letc/passwd xxx2> a.txt/ etc/ passwd
cat a.txt
ls:无法访问xxx:没有那个文件或目录主:使用2>操作符时,会像使用>一样覆盖目标文件的内容,若追加而不覆盖文件的内容即可使用2>>操作符
管道操作
管道(pipe)操作为不同命令之间的协同工作提供了一种机制,位于管道符号""左侧的命令输出的结果,将作为右侧命令的输入(处理对象),同一行命令中可以使用多个管道。
在 shell 脚本应用中,管道操作通常用来过滤所需要的关键信息。
$bash
表
示
系
统
提
示
符
,
表示系统提示符,
表示系统提示符,表示此用户为普通用户,超级用户的提示符是#,bash是shell的一种,是linux下最常用的一种shell
b
a
s
h
的
意
思
是
执
行
一
个
子
s
h
e
l
l
,
此
子
s
h
e
l
l
为
b
a
s
h
。
列
1
:
r
p
m
−
q
a
l
g
r
e
p
h
t
t
p
d
g
r
e
p
"
/
b
i
n
/
b
a
s
h
bash的意思是执行一个子shell,此子shell为bash。 列1: rpm -qa l grep httpd grep "/bin/bash
bash的意思是执行一个子shell,此子shell为bash。列1:rpm−qalgrephttpdgrep"/bin/bash" /etc/passwd / awk -F: “{ print $1,$7 }”
例2
df -Th l grep "/$” | awk “{print $6}”
shell变量及赋值
shell的变量
变量是用来临时保存数据的,并且该数据时可以变化的,任何一个语言都离不开变量,如果某个内容需要多次使用并且会重复出现,这样就可以使用变量了,如果需要修改直接修改变量就可以了
常见 shell变量的类型包括自定义变量、环境变量、只读变量、位置变量、预定义变量
自定义变量
1、变量的定义
Bash中的变量操作相对比较简单,不像其他高级编程语言(如c/C++、Java等)那么复杂。在定义一个新的变量时,一般不需要提前进行声明,而是直接指定变量名称并赋给初始值(内容)即可
格式:变量名=变量值
变量名:临时存放数据的地方变量值:临时的可变化的数据
等号两边没有空格。变量名称需以字母或下划线开头,名称中不要包含特殊字符(如+、-、*、/、 .、 ?、 %、&、#等)
用echo查看和引用变量的值
通过在变量名称前添加前导符号"$”,可以引用一个变量的值,使用echo命令可以查看变量,可以在一条echo命令中同时查看多个变量值
例1
Product=Python
version=2.7.13
echo $Productsversion
当变量名称容易和紧跟其后的其他字符相混淆时,需要添加大括号"{(}"将其括起来,否则将无法确定正确的变量名称。对于未定义的变量,将显示为空值
举例说明
例2
引用变量
echo ${ Product } 2.5
echo ${test }RMB
echo选项
echo -n表示不换行输出
使用echo-e输出转义字符,将转义后的内容输出到屏幕上
常用的转义字符如下:
ic不换行输出,在"\c”后而不存在字符的情况下,作用相当于echo -nkn换行
%t转义后长示入tab.即制表符
注:\转义符,跟在\之后的特殊符号将失去特殊含义,变为普通字符。如$将输出""$"符号,而不当做是变量引用
例1
[ root8localhost ~].echo -n hellohello
[ rootelocalhost ~].
例2
[ rootelocalhost ~]#echo -el"helloit"hello
例3 转义符案例:
echo $VAR1
echo $VAR1
$VAR1
取消定义
unset变量名
特殊操作
还有一些特殊的赋值操作,可以更灵活地为变量赋值,以便适用于各种复杂的管理任务
双引号(" )
双引号主要起界定字符串的作用,特别是当要赋值的内容中包含空格时,必须以双引号括起来;其他情况下双引号通常可以省略
当内容中有空格echo "“hello world”
echo nihao
单引号( ')
当要赋值的内容中包含$、"、\等具有特殊含义的字符时,应使用单引号括起来。
在单引号的范围内,将无法引用其他变量的值,任何字符均作为普通字符看待。输入什么就显示什么但赋值内容中包含单引号(·)时,需使用’符号进行转义,以兔冲突。
[ rootelocalhost ~]# test=123
[rootelocalhost ~]# echo "$test"
123
[ rootelocalhost ~]# echo '$test'
$test
反撇号(`)
反撇号主要用于命令替换,允许将执行某个命令的屏幕输出结果赋值给变量。反撇号括起来的范围内必须是能够执行的命令行,否则将会出错
ls -lh which useradd
$( )
使用反撇号难以在一行命令中实现嵌套命令替换操作,这时可以改用 $( ) 来代替反撇号操作、以解决嵌套的问题
rpm -qc $(rpm -qf $ ( which useradd) )
read命令
除了上述赋值操作以外,还可以使用 Bash的内置命令 read来给变量赋值。
用来提示用户输入信息,从而实现简单的交互过程。执行时将从标准输入设备(键盘)读入一行内容,并以空格为分隔符,将读入的各字段依次赋俏给指定的变量(多余的内容赋值给最后一个变量)。若指定的变量只有一个,则将整行内容赋值给此变量。
[root@localhost ~]# read test
123 //等待用户输入,把输入的值赋于tesp变量
[ root@localhost ~]#echo $test
123
一般来说为了使交互式操作的界面更加友好,提高易用性, read命令可以结合"-p"选项来设置提示信息,以使告知用户应该输入什么内容等相关事项
[rootelocalhost ~] # read -p "请输入你的姓名: " name
请输入你的姓名: ykx
[ rootelocalhost ~]# echo $name
ykx
交互式定义变量(read)
-p提示用广的信息
-n定义字符数
-s不显示用户输入的内容,常用于输入密码
read -s -p "input your password: " pass
-t定义超时时间,超过多长时回没输自动退出
从文件读取内容赋值给变量
[root@server myscripts]# echo 192.168.100.100 > ip.txt
[ root@server myscripts]# cat ip.txt
192.168.100.100
[root@server myscripts]# read -p "input your ip:" IP < ip.txt
[ rooteserver myscriptsj:echo $IP
192.168.100.100
stty -echo/关闭屏幕回显
stty echo/开启屏幕回显
变量的作用范围
默认情况下,新定义的变量只在当前的 shell环境中有效,因此称为局部变量,当进入了程序或新的了shell环境时。局部变量将无法再使用
[ rootelocalhost ~]# bash进入了shell环境
[ rootelocalhost ~]#
[ rootelocalhost ~]# echo $name //不显示变量
export命令
为了使用户定义的变量在所有的了 shell环境中能够继续使用,减少重复设置工作,可以通过内部命令export
将指定的变量导出为全局变量。用户可以同时指定多个变量名称作为参数(无须使用"$"符号),变量名之间以空格分隔
使用export导出全局变量的同时,也可以为变量进行赋值,这样在新定义全局变量时就不需要提前进行赋值
env查看用户当前环境变量
export ABC=123
再次env就能看到了
export -n ABC取消定义的全局变量变成局部变量
数值变量的运算及特殊变量
在 Bash shell环境中,只能进行简单的整数运算,不支持小数运算整数值的运算主要通过内部命令expr进行
运算符与变量之间必须有至少一个空格。
运算内容:加(+)、减(一)、乘()、除( / )、取余(%)运算符号:
(
(
)
)
和
(( ))和
(())和[ ]
运算命令:expr和let
运算工具: bc(系统自带)
expr命令(不仅可以运算,还支持输出到屏幕)常用的几种运算.符如下所述。
+:加法运算。
-:减法运算。
*:乘法运算,注意不能仅使用""符号,否则将被当成文件通配符。
/ :除法运算。
%:求模运算,又称为取余运算,用来计算数值相除后的余数。
例1
[root@server myscripts]# expr 1 + 12
[ root@server myscripts]# expr 1 + 11 + 1
[ root@server myscripts]# expr 2 * 2
expr: i语法错误
[root@server myscripts]# expr 2 \* 2
4
[root@server myscripts]# expr 2 '*' 2
4
乘法也可以用单引号表示但没太大必要因为只有一个字符
两数相加脚本
#!/bin/bash
#1.定义输出数字
read -p "请输入第一个数字:" num1
read -p "请输入第二个数字:" num2
#2.执行加法运算
sum=`expr $num1 + $num2`
echo "求和数:$sum"
let的运算可以改变变量本身的值,但不显示结果,需要echo,其他的运算方式可以做运算但不改变变量本身的值
[ ] 和 []和 []和(())必须要和echo在一起用因为他只能运算无法输出结果
[root@server myscripts]# echo$((1+1))
2
[root@server myscripts]# echo $((5-2))
3
[root@client opt]# echo $((a-b))
7
[root@client opt]f echo $((b-a)) //可以有负数
-7
$[]整数运算
[root@server myscripts]# echo $[10*10] //$[]里的*不需要转义
100
[rooteserver myscripts]# echo $[10%8]
2
[rooteserver myscripts]# echo $[10/8]
1
[rooteserver myscripts]# echo $[10/12]
0
[rooteserver myscripts]# echo $[10%12]
10
[ ] 变 量 的 运 算 , 可 省 略 [ ] 里 的 [ ]变量的运算,可省略[]里的 []变量的运算,可省略[]里的
[root@client opt]# echo $[$a+$b]
13
[root@client opt]# echo $[a+b]
13
[root@client opt]# echo $[a-b]
7
[root@client opt]# echo $[a*b]
30
[rooteclient opt]#echo $ [a/b]
5
使用bc进行运算,支持小数运算,但在脚本中不可直接使用否则会进入交互界面,可以用echo结合管道使用
[rootelocalhost ~]# bc
bc 1.b6.95
copyright 1991-1994,1997,1998,2000,2004,2006 Free software Foundation,Inc.This is free software with ABSOLUTELY NO WARRANTY.
For details type 'warranty' .
10/3
3
scale=3 //指定小数点后几位
10/3
3.333
[root@localhost ~]# echo "scale=3;10/3" | bc
3.33310
[root@client ~]# echo "3^2" | bc //做幂的运算,计算3的平方
9
bc做变量的运算:
[root@1ocalhost ~]# a=10
[root@localhost ~]# b=3
[root@1ocalhost ~]# echo "$a/ $b" | bc
3
[root@localhost ~]# echo "scale=2; $a/ $b" | bc
3.33
bc还可以做逻辑运算,真为1,假为0
[roote@client ~]# echo "2>2" | bc
0
[rootclient ~]# echo "2==2" | bc
1
[rootclient ~]# echo "2<2" | bc
0
常用的运算表达式:
i=$(expr 12 \* 5)
i=$((12 * 5))
i=[12 * 5]
let i=12*5
i++相当于i=$[$i+1]
i--相当于i=$[$i-1]
i+=2相当于i=$[$i+2]
环境变量
环境变量指的是出于运行需要而由Linux系统提前创建的一类变量,主要用于设置用户的工作环境,包括用户宿主日录、命令查找路径、用户当前目录、登录终端等。
环境变量的值由Linux系统自动维护,会随着用户状态的改变而改变。
使用env命令可以查看到当前工作环境下的环境变量,对于常见的一些环境变量应了解其各自的用途。
例如,变量USER表示用户名称,HOME 表示用户的宿主目录,LANG表示语言和字符集,PWD表示当前所在的工作日录,PATH
表示命令搜索路径等、RANDOM表示随机数,会返回0-32767的整数,USER表示当前账户的账户名称等,一般都用全大写定义,注意和自定义变量区分
[ root@localhost ~]#echo $RANDOM
20315
[ root@localhost ~]#echo $HOME
/root
[ root@localhost ~]#echo $USER
root
[ rooteiocalhost ~]#echo $UID
0
[ rootelocalhost ~]#echo $PWD
/root
[ rootelocalhost ~]#echo $LANG
zh_cN. UTF-8
[ rootelocalhost ~]#echo $SHEL
/bin/bash
[rooteclient ~]# echo $RANDOM //linux中$random用于生成0-32767的随机数
29780
PATH 变量用于设置可执行程序的默认搜索路径,当仅指定文件名称来执行命令程序时,Linux系统将在 PATH
变量指定的目录范围查找对应的可执行文件,如果找不到则会提示"command not found”。
[root@localhost ~]# test.sh
bash: test.sh:未找到命令...
[root@localhost ~]#echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin
[root@localhost ~]#pwd
/root
这时因为test.sh不在$PATH的目录里面,所以系统无法识别无法直接用,需要跟上绝对路径使用该脚本
方法一:将脚本的目录加入$PATH
[root@localhost -]# PATH="$PATH:/root" //这是临时的,如果永久生效需要编辑/etc/profile文件
[root@1ocalhost ~]# echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin:/root
[root@localhost ~]# test.sh
hello world
方法二:将你自己写的脚本放到$PATH中的某一个目录
在 Linux系统中,环境变量的全局配置文件为letc/profile,在此文件中定义的变量作用于所有用户。除此之外,每个用户还有自己的独立配置文件(~/.bash_profile)。
修改完了要重新登陆才能生效,如果想立即生效,可以使用source
注意:修改$PATH需要慎重操作,如果找不到了会影响命令的使用! ! !
例如:
[root@localhost ~]#PATH= //手误将PATH设为空
[root@localhost ~]#echo $PATH
[root@localhost ~]#ls
-bash: ls:没有那个文件或目录
位置变量
当执行命令行操作时,第一个字段表示命令名或脚本程序名,其余的字符串参数按照从左到右的顺序依次赋值给位置变量。
位置变量也称为位置参数,使用$1、$2、$3、…、$9表示
命令或脚本本身的名称使用"$0"表示
例1:编写一个简单的创建用户和密码的脚本
[root@localhost ~]#vim user.sh
#!/bin/bash
useradd $1
echo $2 | passwd --stdin $1
[root@1ocalhost ~]#bash user.sh sj123
更改用户 sj的密码。
passwd:所有的身份验证令牌已经成功更新。
[root@localhost ~]#
[root@localhost ~]#id sj
uid=1001(sj) gid=1001(sj) 组=1001(sj)
例2
[root@localhost~]#vim adder2num.sh
#!/bin/bash
SUM=`expr $1 +$2`
echo "$1 + $2 = $sUM"
[root@localhost ~]#chmod +x adder2num.sh
[root@localhost ~]#./adder2num.sh 12 34 //$1为12、$2为34的情况12+34=46
[root@localhost ~]#./adder2num.sh 56 78 //$1为56、$2为78的情况56+78=134
预定义变量
预定义变量是由Bash程序预先定义好的一类特殊变量,用户只能使用预定义变量,而不能创建新的预定义变量,也不能直接为预定义变量赋值。预定义变量使用"$"符号和另一个符号组合表示
$#:表示命令行中位置参数的个数。
$*:表示所有位置参数的内容,这些内容当做一个整体
$@:表示列出所有位置参数,但是是以单个的形式的列出
$?:表示前一条命令执行后的返回状态,返回值为0表示执行正确,返回任何非0值均表示执行出现异常。
$0:表示当前执行的脚本或程序的名称
$$:表示返回当前进程的进程号
$!:返回最后一个后台进程的进程号
例1:根据一个简单的脚本来理解每个预定义和位置变量的含义
#!/bin/bash
echo $1
echo “
0
表
示
当
前
执
行
的
脚
本
或
程
序
的
名
称
"
e
c
h
o
"
0表示当前执行的脚本或程序的名称" echo "
0表示当前执行的脚本或程序的名称"echo"#表示命令行中位置参数的个数”
echo “
∗
所
有
位
置
参
数
的
内
容
,
这
些
内
容
当
做
一
个
整
体
”
e
c
h
o
"
*所有位置参数的内容,这些内容当做一个整体” echo "
∗所有位置参数的内容,这些内容当做一个整体”echo"@表示列出所有位置参数,但是是以单个的形式的列出”
例2:可以将脚本做下优化
#!/bin/ bash
echo “当前脚本名称为$0”
echo “当前脚本的第一个参数是$1”
echo “当前脚本的第二个参数是
2
"
e
c
h
o
"
当
前
脚
本
一
共
有
2" echo "当前脚本一共有
2"echo"当前脚本一共有#个参数”
例3 touch /home/ kgc.txt &
echo “$
表
示
返
回
当
前
进
程
的
进
程
号
"
e
c
h
o
"
表示返回当前进程的进程号" echo "
表示返回当前进程的进程号"echo"? 0正确,1”
echo “$!返回最
后一个后台进程的进程号”
总结:
重定向与管道操作是shell
环境中十分常用的功能,若能够熟练掌握并灵活运用,将有助于编写代码简洁但功能强大的shell脚本程序