找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 144|回复: 0

Vim编辑器与Shell命令脚本

[复制链接]

983

主题

83

回帖

5万

积分

管理员

积分
53122
发表于 2022-6-23 10:18:57 | 显示全部楼层 |阅读模式
章节简述:
! [% s: f& j/ {本章首先介绍如何使用Vim编辑器来编写和修改文档,然后通过逐步配置主机名称、系统网卡以及软件仓库等文件,帮助大家加深Vim编辑器中诸多命令、快捷键与模式的理解。然后会带领大家重温第2章和第3章中的重点知识,做到Linux命令、逻辑操作符与Shell脚本的灵活搭配使用。
) n$ t2 y3 b# i# {7 F本章还要求大家能够在Shell脚本中以多种方式接收用户输入的信息,能够对输入值进行文件、数字、字符串的判断比较。在熟练使用“与、或、非”三种逻辑操作符的基础上,大家还要充分学习if、for、while、case条件测试语句,并通过10多个实战脚本的实操练习,达到在工作中灵活运用的水准。
! G1 @/ b7 y' O( g" Y1 c+ B本章最后通过实战的方式演示了使用at命令与crond计划任务服务来分别实现一次性的系统任务设置和长期性的系统任务设置,在分钟、小时、日期、月份、年份的基础上实现工作的自动化,从而让日常的工作更加高效,可以让大家早点下班陪孩子。. j  B# F3 l* G% W
本章目录结构
  • 4.1 Vim文本编辑器2 K& L) w" o% I. {! X
    • 4.1.1 编写简单文档
    • 4.1.2 配置主机名称
    • 4.1.3 配置网卡信息
    • 4.1.4 配置软件仓库
      % ?$ {3 ^3 p5 P) K# G  S
  • 4.2 编写Shell脚本1 n1 y9 }) b4 p  @, h8 D
    • 4.2.1 编写简单的脚本
    • 4.2.2 接收用户的参数
    • 4.2.3 判断用户的参数
      " k  ^2 I$ C* c0 K
  • 4.3 流程控制语句; U& \9 K: z' n+ e0 k9 U
    • 4.3.1 if条件测试语句
    • 4.3.2 for条件循环语句
    • 4.3.3 while条件循环语句
    • 4.3.4 case条件测试语句
      7 \+ J, C! z4 S% l: M; j
  • 4.4 计划任务服务程序8 M9 ~( Y0 W! b# Q
: V3 M/ r' f/ S( W, v2 W
4.1 Vim文本编辑器Vim的发布最早可以追溯到1991年,英文全称为Vi Improved。它也是Vi编辑器的提升版本,其中最大的改进当属添加了代码着色功能,在某些编程场景下还能自动修正错误代码。
$ m$ u% f+ _$ W+ U' K每当在讲课时遇到需要让学生记住的知识点时,为了能让他们打起精神来,我都会突然提高嗓门,因此有句话他们记得尤其深刻:“在Linux系统中一切都是文件,而配置一个服务就是在修改其配置文件的参数。”而且在日常工作中大家也肯定免不了要编写文档,这些工作都是通过文本编辑器来完成的。刘遄老师写作本书的目的是让读者切实掌握Linux系统的运维方法,而不是仅仅停留在“会用某个操作系统”的层面上,所以我们这里选择使用Vim文本编辑器,它默认会安装在当前所有的Linux操作系统上,是一款超棒的文本编辑器。$ {' s  j1 |' R
Vim之所以能得到广大厂商与用户的认可,原因在于Vim编辑器中设置了3种模式—命令模式、末行模式和编辑模式,每种模式分别又支持多种不同的命令快捷键,这大大提高了工作效率,而且用户在习惯之后也会觉得相当顺手。要想高效地操作文本,就必须先搞清这3种模式的操作区别以及模式之间的切换方法(见图4-1)。
7 S# C3 g: @. ~) V6 q9 g1 `' F
命令模式:控制光标移动,可对文本进行复制、粘贴、删除和查找等工作。
# X7 `% [1 i9 d: k输入模式:正常的文本录入。3 ~9 W  l- I6 {$ ]( p# {2 R
末行模式:保存或退出文档,以及设置编辑环境。
! u2 S! |7 I, f) O8 [/ N  g! e+ S5 {
图4-1  Vim编辑器模式的切换方法
在每次运行Vim编辑器时,默认进入命令模式,此时需要先切换到输入模式后再进行文档编写工作。而每次在编写完文档后需要先返回命令模式,然后再进入末行模式,执行文档的保存或退出操作。在Vim中,无法直接从输入模式切换到末行模式。Vim编辑器中内置的命令有成百上千种用法,为了能够帮助读者更快地掌握Vim编辑器,表4-1总结了在命令模式中最常用的一些命令。5 G/ W% o: [$ T( Z! V6 G3 D
表4-1                                                    命令模式中最常用的一些命令
& O' p" H2 i% b* H' V$ y. p
命令作用
dd删除(剪切)光标所在整行
5dd删除(剪切)从光标处开始的5行
yy复制光标所在整行
5yy复制从光标处开始的5行
n显示搜索命令定位到的下一个字符串
N显示搜索命令定位到的上一个字符串
u撤销上一步的操作
p将之前删除(dd)或复制(yy)过的数据粘贴到光标后面
末行模式主要用于保存或退出文件,以及设置Vim编辑器的工作环境,还可以让用户执行外部的Linux命令或跳转到所编写文档的特定行数。要想切换到末行模式,在命令模式中输入一个冒号就可以了。末行模式中常用的命令如表4-2所示。
* l$ Y8 A& z% E表4-2                                                   末行模式中最常用的一些命令3 x# G" [% R; x4 ^
命令作用
:w保存
:q退出
:q!强制退出(放弃对文档的修改内容)
:wq!强制保存退出
:set nu显示行号
:set nonu不显示行号
:命令执行该命令
:整数跳转到该行
:s/one/two将当前光标所在行的第一个one替换成two
:s/one/two/g将当前光标所在行的所有one替换成two
:%s/one/two/g将全文中的所有one替换成two
?字符串在文本中从下至上搜索该字符串
/字符串在文本中从上至下搜索该字符串
大家在平日里一定要多使用Vim编辑器,一旦把Vim的各种命令练熟,后面在编辑配置文件时,效率就会有很大的提升。在2011年,有一位名为Aleksandr Levchuk的极客,他就为了追求极致的效率,发起了一个名为VIM Clutch的实验项目。他买了一对类似于汽车油门和刹车的离合器,改装后再用USB与电脑相连,左脚踩刹车是进入编辑模式(i),右脚踩油门是保存文件(wq!)。他对Linux和Vim的热爱真是强大!! \. b! O' N+ e6 ^! c
4.1.1 编写简单文档目前为止,大家已经具备了在Linux系统中编写文档的理论基础,接下来我们一起动手编写一个简单的脚本文档。刘遄老师会尽力把所有操作步骤和按键过程都标注出来,如果忘记了某些快捷键命令的作用,可以再返回前文进行复习。8 {  ?8 B7 O5 Y* V& Z/ D3 X3 e
编写脚本文档的第1步就是给文档取个名字,这里将其命名为practice.txt。如果存在该文档,则是打开它。如果不存在,则是创建一个临时的输入文件,如图4-2所示。8 z1 b& p3 ~, P/ J9 b
图4-2  尝试编写文本文档
打开practice.txt文档后,默认进入的是Vim编辑器的命令模式。此时只能执行该模式下的命令,而不能随意输入文本内容,我们需要切换到输入模式才可以编写文档。
, d. q$ n5 [8 x! Z在图4-1中提到,可以分别使用a、i、o三个键从命令模式切换到输入模式。其中,a键与i键分别是在光标后面一位和光标当前位置切换到输入模式,而o键则是在光标的下面再创建一个空行,此时可敲击a键进入到编辑器的输入模式,如图4-3所示。
0 d* B% k1 G" o1 a8 N
图4-3  切换至编辑器的输入模式
进入输入模式后,可以随意输入文本内容,Vim编辑器不会把您输入的文本内容当作命令而执行,如图4-4所示。8 W8 B3 W: t! Z6 j- ~4 ?/ o
图4-4  在编辑器中输入文本内容
在编写完之后,要想保存并退出,必须先敲击键盘的Esc键从输入模式返回命令模式,如图4-5所示。然后再输入“:wq!”切换到末行模式才能完成保存退出操作,如图4-6所示。: A% }2 d. x) i9 }) Z4 Q2 s+ }7 J
图4-5 切换至编辑器的命令模式
图4-6 切换至编辑器的末行模式
Tips
, v* T# G4 @" J1 X- b! S6 I请各位同学仔细观察图4-4~图4-6中左下角的提示信息,在不同模式下有不同的提示字样。  s6 \8 {5 a1 ?* K! _( D9 F! b

: Q; |- H- |, Q/ o当在末行模式中输入“:wq!”命令时,就意味着强制保存并退出文档。然后便可以用cat命令查看保存后的文档内容了,如图4-7所示。! I; _9 q# F7 t) k
图4-7 查看文档的内容
是不是很简单?!继续编辑这个文档。因为要在原有文本内容的下面追加内容,所以在命令模式中敲击o键进入输入模式更会高效,操作如图4-8~图4-10所示。6 Y9 a- p4 T1 T
图4-8  再次通过Vim编辑器编写文档
图4-9  进入Vim编辑器的输入模式
图4-10  追加写入一行文本内容
因为此时已经修改了文本内容,所以Vim编辑器在我们尝试直接退出文档而不保存的时候就会拒绝我们的操作了。此时只能强制退出才能结束本次输入操作,如图4-11~图4-13所示。
" V/ k  v" s5 k; ~0 L6 `6 I
图4-11  退出文本编辑器
图4-12  因文件已被修改而拒绝退出操作
图4-13  强制退出文本编辑器
现在大家也算是具有了一些Vim编辑器的实战经验了,应该也感觉到没有想象中那么难吧。现在查看文本的内容,果然发现追加输入的内容并没有被保存下来,如图4-14所示。
' K# L, x0 J$ j! q  B( q
图4-14  查看最终编写成的文本内容
大家在学完了理论知识之后又自己动手编写了一个文本,现在是否感觉成就满满呢?接下来将会由浅入深地为读者安排3个小任务。为了彻底掌握Vim编辑器的使用,大家一定要逐个完成不许偷懒,如果在完成这3个任务期间忘记了相关命令,可返回前文进一步复习掌握。  V- q- e! b2 _8 P6 \
Tips
5 G& h0 [6 d* Z4 ^8 c下面的实验如果做不成功也很正常,请大家把重心放到Vim编辑器上面,能成功修改配置文件就已经很棒啦!
8 e( E, P0 b; l1 q8 M" D+ l/ ]0 F! R! W
4.1.2 配置主机名称为了便于在局域网中查找某台特定的主机,或者对主机进行区分,除了要有IP地址外,还要为主机配置一个主机名,主机之间可以通过这个类似于域名的名称来相互访问。在Linux系统中,主机名大多保存在/etc/hostname文件中,接下来将/etc/hostname配置文件的内容修改为“linuxprobe.com”,步骤如下。+ T/ D( V# N1 k5 X3 O
第1步:使用Vim编辑器修改/etc/hostname主机名称文件。4 ^( [, [/ A- @3 {8 E3 o) E
第2步:把原始主机名称删除后追加“linuxprobe.com”。注意,使用Vim编辑器修改主机名称文件后,要在末行模式下执行“:wq!”命令才能保存并退出文档。
9 E4 _. }' p2 T+ t2 K9 d' P第3步:保存并退出文档,然后使用hostname命令检查是否修改成功。
% T# S  ]+ ^# Z; ]! @. h( W* q- A

2 K  x3 p! A, ^* S" I' Q, i
  1. [root@linuxprobe ~]# vim /etc/hostname
    ; ~" ^" S. t2 ~* _
  2. linuxprobe.com- a) z; g& U2 I4 o) @+ d3 }
  3. 1 g4 w0 U  @8 V8 e' }
  4. hostname命令用于查看当前的主机名称,但有时主机名称的改变不会立即同步到系统中,所以如果发现修改完成后还显示原来的主机名称,可重启虚拟机后再行查看:' ]& w* j0 b1 @3 _% d, [& l
  5. 2 ]! R* s1 k' k- @0 t/ _! y- E
  6. [root@linuxprobe ~]# hostname
    ( X* V5 K3 H( `5 r
  7. linuxprobe.com
    * A* d" Q- Y/ C/ C" |
复制代码
' y2 b& m2 L# c
0 v: l) Q0 D/ F  p, q/ i6 `! @
4.1.3 配置网卡信息网卡IP地址配置的是否正确是两台服务器是否可以相互通信的前提。在Linux系统中,一切都是文件,因此配置网络服务的工作其实就是在编辑网卡配置文件。这个小任务不仅可以帮助您练习使用Vim编辑器,而且也为后面学习Linux中的各种服务配置打下了坚实的基础。当您认真学习完本书后,一定会特别有成就感,因为本书前面的基础部分非常扎实,而后面内容则具有几乎一致的网卡IP地址和运行环境,从而确保您全身心地投入到各类服务程序的学习上,而不用操心系统环境的问题。
0 ~9 p  `: m  B: b如果您具备一定的运维经验或者熟悉早期的Linux系统,则在学习本书时会遇到一些不容易接受的差异变化。在RHEL 5、RHEL 6中,网卡配置文件的前缀为eth,第1块网卡为eth0,第2块网卡为eth1;以此类推。在RHEL 7中,网卡配置文件的前缀则以ifcfg开始,再加上网卡名称共同组成了网卡配置文件的名字,例如ifcfg-eno16777736。而在RHEL 8中,网卡配置文件的前缀依然为ifcfg,区别是网卡名称改成了类似于ens160的样子,不过好在除了文件名发生变化外,网卡参数没有其他大的区别。
9 j' n6 g2 A  {/ ^; k9 ^$ |+ q  y现在有一个名称为ifcfg-ens160的网卡设备,将其配置为开机自启动,并且IP地址、子网、网关等信息由人工指定,其步骤如下所示。
2 _! @, t9 P) o; H# v, U4 X第1步:首先切换到/etc/sysconfig/network-scripts目录中(存放着网卡的配置文件)。
9 K3 ]3 A5 i. t* }第2步:使用Vim编辑器修改网卡文件ifcfg-ens160,逐项写入下面的配置参数并保存退出。由于每台设备的硬件及架构是不一样的,因此请读者使用ifconfig命令自行确认各自网卡的默认名称。8 q3 a# J9 p; Q( {+ c2 ^) Z* B6 D5 Z
  1. 设备类型:TYPE=Ethernet
    6 o3 \" L1 d4 F+ V- s8 s
  2. 地址分配模式:BOOTPROTO=static
    . [: D# I/ v+ v% C/ a: Q3 W
  3. 网卡名称:NAME=ens1602 d$ }' c/ L! Q9 E. I% ?+ E/ Y
  4. 是否启动:ONBOOT=yes6 F9 m- z" K7 @; G9 C
  5. IP地址:IPADDR=192.168.10.10
    $ i( c% f/ X) [  D
  6. 子网掩码:NETMASK=255.255.255.0$ }& w: c# n) j$ p
  7. 网关地址:GATEWAY=192.168.10.1' Z) x/ M" F6 p( J3 b) v
  8. DNS地址:DNS1=192.168.10.1
复制代码

9 E% e0 w  ^- M7 Q% p3 {8 R5 y+ Z' r, g, r% s
第3步:重启网络服务并测试网络是否连通。* |1 b2 j; t7 x/ i* ^( E3 V
下面正式开干!
  H4 a4 o+ E: ?! _+ s, Y进入到网卡配置文件所在的目录,然后编辑网卡配置文件,在其中填入下面的信息:
( C( j  O1 N' d0 l, `- @4 Z* @
  1. [root@linuxprobe ~]# cd /etc/sysconfig/network-scripts/
    / G. t* E& f  {: r' I- }
  2. [root@linuxprobe network-scripts]# vim ifcfg-ens1608 l; }, P, `+ A6 h1 Q$ ^' h
  3. TYPE=Ethernet+ d8 g5 g% r$ S3 j
  4. BOOTPROTO=static
    ' w- o; m) y: G. |( {. ]
  5. NAME=ens160
    + C4 Y* i5 Z& g3 D5 U
  6. ONBOOT=yes
    5 e4 g5 |; ?8 p% }/ L' |
  7. IPADDR=192.168.10.10
    5 J2 s! r4 R+ k2 a
  8. NETMASK=255.255.255.05 G! v3 x' ^9 n; D9 \6 X
  9. GATEWAY=192.168.10.1
    $ s2 M+ J8 {2 v- s
  10. DNS1=192.168.10.1
复制代码
. Z, V+ c, l3 U0 c

2 i6 v- Q* x- H9 J% G执行重启网卡设备的命令,然后通过ping命令测试网络能否连通。由于在Linux系统中ping命令不会自动终止,因此需要手动按下Ctrl+C组合键来强行结束进程。
8 G, u# }) F$ T: l  o) a
  1. [root@linuxprobe network-scripts]# nmcli connection reload ens160
    * ]4 Y' A- K5 c; K/ W, ?0 `
  2. [root@linuxprobe network-scripts]# ping 192.168.10.10' J7 @3 G7 |6 u; v# T
  3. PING 192.168.10.10 (192.168.10.10) 56(84) bytes of data.
    : q! |  y( O8 f7 w8 w# P
  4. 64 bytes from 192.168.10.10: icmp_seq=1 ttl=64 time=0.083 ms4 M8 [/ Z& ~6 O
  5. 64 bytes from 192.168.10.10: icmp_seq=2 ttl=64 time=0.110 ms0 f, z7 _2 G9 Y8 I# \+ h
  6. 64 bytes from 192.168.10.10: icmp_seq=3 ttl=64 time=0.106 ms3 L9 @4 R; p8 G3 R! a
  7. 64 bytes from 192.168.10.10: icmp_seq=4 ttl=64 time=0.035 ms4 J5 y1 E( a# |% I% Z) \1 A
  8. ^C" M. F3 C" p1 J2 d2 B8 N
  9. --- 192.168.10.10 ping statistics ---
    $ q- d0 c% j* Y; s4 D* E3 k0 B
  10. 4 packets transmitted, 4 received, 0% packet loss, time 84ms! f* D& c1 W% m5 I; g0 r6 {
  11. rtt min/avg/max/mdev = 0.035/0.083/0.110/0.031 ms' a, ~6 l  C, G+ O0 E" y8 N( Y' c
  12. [root@linuxprobe network-scripts]#
复制代码
6 d4 Y3 M' u/ V+ D" D
9 I( u/ Y% N0 Z: r
是不是感觉很有意思?!当然如果这个实验失败了也不用气馁,后面会有相应的章节专门讲解,请大家把关注点继续放回到Vim编辑器上就好。
5 X( Y+ g( }4 `$ v* Z4.1.4 配置软件仓库本书前面讲到,软件仓库是一种能进一步简化RPM管理软件的难度以及自动分析所需软件包及其依赖关系的技术。可以把Yum或DNF想象成是一个硕大的软件仓库,里面保存有几乎所有常用的工具,而且只需要说出所需的软件包名称,系统就会自动为您搞定一切。
8 D' r1 X8 U# `! [, b既然要使用软件仓库,就要先把它搭建起来,然后将其配置规则确定好才行。鉴于第6章才会讲解Linux的存储结构和设备挂载操作,所以当前还是将重心放到Vim编辑器的学习上。如果遇到看不懂的参数也不要紧,后面章节会单独讲解。0 H0 T9 C$ g  a% \
Yum与DNF软件仓库的配置文件是通用的,也就是说填写好配置文件信息后,这两个软件仓库的命令都是可以正常使用。建议在RHEL 8中使用dnf作为软件的安装命令,因为它具备更高的效率,而且支持多线程同时安装软件。/ Z* B8 y  _. }6 g, z
搭建并配置软件仓库的大致步骤如下所示。- M; x8 S# i; z  A  b5 U" e4 A
第1步:进入/etc/yum.repos.d/目录中(因为该目录存放着软件仓库的配置文件)。1 D3 V( v) L2 J. H
第2步:使用Vim编辑器创建一个名为rhel8.repo的新配置文件(文件名称可随意,但后缀必须为.repo),逐项写入下面的配置参数并保存退出。
3 I% v* A' H' F6 e
仓库名称:具有唯一性的标识名称,不应与其他软件仓库发生冲突。
' G" b4 o1 ?+ |6 i描述信息(name):可以是一些介绍性的词,易于识别软件仓库的用处。# f: R" O+ T. X% n
仓库位置(baseurl):软件包的获取方式,可以使用FTP或HTTP下载,也可以是本地的文件(需要在后面添加file参数)。! z: ^  N) ~- p% I9 u9 }
是否启用(enabled):设置此源是否可用;1为可用,0为禁用。$ X1 A" Z5 n2 R  B( o& E
是否校验(gpgcheck):设置此源是否校验文件;1为校验,0为不校验。
6 g; E" S' `4 {1 F5 k8 u% ]; H公钥位置(gpgkey):若上面的参数开启了校验功能,则此处为公钥文件位置。若没有开启,则省略不写。% R" {; i& n$ t6 v
第3步:按配置参数中所填写的仓库位置挂载光盘,并把光盘挂载信息写入/etc/fstab文件中。
7 {4 s6 M  D; h! L( W第4步:使用“dnf install httpd -y”命令检查软件仓库是否已经可用。
/ c6 W& Y, z6 \/ L0 x( ~开始实战!
: Q# ?* ^: L3 i进入/etc/yum.repos.d目录后创建软件仓库的配置文件:6 l4 i3 m$ Y$ C9 F8 w
  1. [root@linuxprobe ~]# cd /etc/yum.repos.d/) Q1 q" p$ G2 Q+ e9 g
  2. [root@linuxprobe yum.repos.d]# vim rhel8.repo' C6 F5 K# `; `7 D3 r
  3. [BaseOS]" H1 N. n& m. H# D$ `5 b' S
  4. name=BaseOS
    ) E( @7 T8 x" y7 W1 k% [
  5. baseurl=file:///media/cdrom/BaseOS
    $ M% ~% F/ W7 ~
  6. enabled=1
    6 Y$ P' Z4 ]7 h2 @9 ~# h, I
  7. gpgcheck=0& G1 {9 E$ k. N7 y
  8. [AppStream]
    7 s! }/ d; H. }; z. Z
  9. name=AppStream% B/ G' x2 f7 V% Z" E: M% A& y3 O
  10. baseurl=file:///media/cdrom/AppStream
      W, t  A# O6 [) t7 q3 k' S$ Y
  11. enabled=1
    # y  K9 }; M! h6 d- N% b- N
  12. gpgcheck=0
复制代码

. S) ]: ~( U) E/ c
; D0 @! x- d. b. p创建挂载点后进行挂载操作,并设置成开机自动挂载(详见第6章):& v; c4 G, e% ~1 u9 M8 e
  1. [root@linuxprobe yum.repos.d]# mkdir -p /media/cdrom
    % o$ A$ V% k# \( `& H* v
  2. [root@linuxprobe yum.repos.d]# mount /dev/cdrom /media/cdrom4 F; v- g5 j* \. x5 H$ P" Q( x
  3. mount: /media/cdrom: WARNING: device write-protected, mounted read-only./ z7 }6 E( W3 Y0 n# N4 g) t
  4. [root@linuxprobe yum.repos.d]# vim /etc/fstab  z0 C6 e2 j; k" G- A$ ?1 w. ?
  5. /dev/cdrom /media/cdrom iso9660 defaults 0 0
复制代码
! S' E" q8 X$ m) c1 w
2 y9 B5 ~9 u# D7 I% q# m
尝试使用软件仓库的dnf命令来安装Web服务,软件包名称为httpd,安装后出现“Complete!”则代表配置正确:
/ h7 R5 ~3 H9 J6 T7 m( w7 S. H
  1. <pre>[root@linuxprobe ~]# dnf install httpd -y
    2 J8 f9 B) ^% E
  2. Updating Subscription Management repositories.
    9 b2 o# y' }3 Q+ ?* c$ t
  3. Unable to read consumer identity6 {# C* ^/ D) X
  4. This system is not registered to Red Hat Subscription Management. You can use subscription-manager to register.5 a! n8 h* t/ w6 L- y
  5. AppStream 3.1 MB/s | 3.2 kB 00:00" i& s8 K' M" b% {
  6. BaseOS 2.7 MB/s | 2.7 kB 00:000 d' t" X$ ?+ a3 S6 b* f, }$ h
  7. Dependencies resolved.* Z- W4 r8 V9 S
  8. ………………省略部分输出信息………………+ g" l% `0 M; @9 G
  9. Installed:; t% J4 `! V7 A
  10. httpd-2.4.37-10.module+el8+2764+7127e69e.x86_640 [+ n7 H" y( `" ~/ P# T  b
  11. apr-util-bdb-1.6.1-6.el8.x86_64- a7 h! t$ ~, a2 ]6 o; `8 P' d
  12. apr-util-openssl-1.6.1-6.el8.x86_645 @2 {0 Q$ |! S, V! Z
  13. apr-1.6.3-9.el8.x86_64
    4 S. h$ `' ~) r8 b" |7 g
  14. apr-util-1.6.1-6.el8.x86_64+ t. z& J; U0 W/ E/ L8 L
  15. httpd-filesystem-2.4.37-10.module+el8+2764+7127e69e.noarch, {3 }% \  e' P: y8 N: V
  16. httpd-tools-2.4.37-10.module+el8+2764+7127e69e.x86_64$ B3 l9 j$ U. [" V
  17. mod_http2-1.11.3-1.module+el8+2443+605475b7.x86_64
    : O0 l: [0 m6 @: k* }
  18. redhat-logos-httpd-80.7-1.el8.noarch
    . ^- D, v3 |8 h

  19. 8 Q; ]% S. u+ X, X7 F* V
  20. Complete!</pre>
复制代码
- p8 P/ ]7 n' H3 d5 M" N6 `
对于习惯使用yum命令来安装软件的同学,也不需要有压力,因为您依然可以使用yum install httpd命令来安装软件,只是将dnf替换成yum。可见,RHEL 8版本很好地兼容了用户习惯。; ]# @) A+ S9 [# f) h2 m3 v1 M
4.2 编写Shell脚本
  o/ |4 T) J" B( x4 D6 t可以将Shell终端解释器当作人与计算机硬件之间的“翻译官”,它作为用户与Linux系统内部的通信媒介,除了能够支持各种变量与参数外,还提供了诸如循环、分支等高级编程语言才有的控制结构特性。要想正确使用Shell中的这些功能特性,准确下达命令尤为重要。Shell脚本命令的工作方式有下面两种。
$ b) U, h( y# [( W! p
交互式(Interactive):用户每输入一条命令就立即执行。4 |" K2 `  n- P; ^* G. c: }
批处理(Batch):由用户事先编写好一个完整的Shell脚本,Shell会一次性执行脚本中诸多的命令。
; G8 e- [2 @# f$ G/ n3 ]
在Shell脚本中不仅会用到前面学习过的很多Linux命令以及正则表达式、管道符、数据流重定向等语法规则,还需要把内部功能模块化后通过逻辑语句进行处理,最终形成日常所见的Shell脚本。3 O- {/ X9 i. c/ K. U1 g, D6 K
通过查看SHELL变量可以发现,当前系统已经默认使用Bash作为命令行终端解释器了:& M- k% T* }0 Y: U, x
  1. [root@linuxprobe ~]# echo $SHELL5 L& s" \  ]2 `" N0 h* P
  2. /bin/bash
复制代码
) z! q' a6 ^$ l- Q( T; _

! l8 S- `" S/ I: v* K4.2.1 编写简单的脚本估计读者在看完上文中有关Shell脚本的复杂描述后,会累觉不爱吧。但是,上文指的是一个高级Shell脚本的编写原则,其实使用Vim编辑器把Linux命令按照顺序依次写入到一个文件中,就是一个简单的脚本了。1 e2 a0 ^! t$ G* r1 ~2 J5 Z! Q; i2 s9 G
例如,如果想查看当前所在工作路径并列出当前目录下所有的文件及属性信息,实现这个功能的脚本应该类似于下面这样:
+ i* b! h5 g3 _1 `
  1. [root@linuxprobe ~]# vim example.sh
      |1 e( r# J7 h
  2. #!/bin/bash 6 s2 U2 [- ^' y$ J( R
  3. #For Example BY linuxprobe.com
    ( J' v! w+ k) y% F! _) m: @6 {& C$ f6 z5 ?
  4. pwd # @/ v4 T9 t5 [2 J2 M8 X
  5. ls -al
复制代码

& t8 F  I; h+ T2 v, a; b: K% K. Q1 l/ F7 n+ }2 R1 c: x$ L
Shell脚本文件的名称可以任意,但为了避免被误以为是普通文件,建议将.sh后缀加上,以表示是一个脚本文件。9 d  x" _9 n8 E/ T
在上面的这个example.sh脚本中实际上出现了3种不同的元素:第一行的脚本声明(#!)用来告诉系统使用哪种Shell解释器来执行该脚本;第二行的注释信息(#)是对脚本功能和某些命令的介绍信息,使得自己或他人在日后看到这个脚本内容时,可以快速知道该脚本的作用或一些警告信息;第三、四行的可执行语句也就是我们平时执行的Linux命令了。你们不相信这么简单就编写出来了一个脚本程序?!那我们来执行一下看看结果:
; L. n8 _( p5 y4 N. m' M
  1. [root@linuxprobe ~]# bash example.sh1 b( r* ^+ T0 A! b+ r/ J# G  M4 i2 u
  2. /root" P- |4 M; B2 N6 a6 @. ]
  3. total 609 o. ~1 z3 p# k
  4. dr-xr-x---. 15 root root  4096 Oct 12 00:41 .
    * A" o. [. [. T4 w% N
  5. dr-xr-xr-x. 17 root root   224 Jul 21 05:04 ..) j. L1 }. v; ^9 e. n! Q$ @7 Q
  6. -rw-------.  1 root root  1407 Jul 21 05:09 anaconda-ks.cfg
    3 L/ A! Q. _5 V& c# c; {
  7. -rw-------.  1 root root   335 Jul 24 06:33 .bash_history
    $ ]5 ?& a+ G. N9 x/ z. r# E6 a! u
  8. -rw-r--r--.  1 root root    18 Aug 13  2018 .bash_logout4 [" m9 @2 q+ _
  9. -rw-r--r--.  1 root root   176 Aug 13  2018 .bash_profile6 X- }8 E9 r$ s
  10. ………………省略部分输出信息………………
复制代码

' ~1 |3 y$ W- t2 S! S, p
! j: m0 h7 r3 a7 o1 y) U除了上面用Bash解释器命令直接运行Shell脚本文件外,第二种运行脚本程序的方法是通过输入完整路径的方式来执行。但默认会因为权限不足而提示报错信息,此时只需要为脚本文件增加执行权限即可(详见第5章)。初次学习Linux系统的读者不用心急,等下一章学完用户身份和权限后再来做这个实验也不迟:
5 `0 c. T# e3 M' ?+ g/ z# ^
  1. [root@linuxprobe ~]# ./example.sh+ U( y0 o  f: C$ V
  2. bash: ./Example.sh: Permission denied
    " ^( v5 |2 D) G1 K- @
  3. [root@linuxprobe ~]# chmod u+x example.sh
    7 A6 |/ }9 z5 C) k& {; R
  4. [root@linuxprobe ~]# ./example.sh
    - [1 n" C$ P* ~3 ]# x* E- J$ j. F' v
  5. /root
    " [/ N8 c' C& q
  6. total 60& ~9 j, P  n# E* _* |
  7. dr-xr-x---. 15 root root  4096 Oct 12 00:41 .
      R; U/ z+ P. ~
  8. dr-xr-xr-x. 17 root root   224 Jul 21 05:04 ..
    ' ~7 ]6 V- p' `) w( K7 ?- `) }* @
  9. -rw-------.  1 root root  1407 Jul 21 05:09 anaconda-ks.cfg2 x7 I% v2 R) o/ \0 G1 I. i) e
  10. -rw-------.  1 root root   335 Jul 24 06:33 .bash_history
    : f( |# O  z* G
  11. -rw-r--r--.  1 root root    18 Aug 13  2018 .bash_logout4 _. M+ U6 z  _2 t4 Z5 C' ?: A
  12. -rw-r--r--.  1 root root   176 Aug 13  2018 .bash_profile9 X6 E: c! s$ {+ E/ n1 B
  13. ………………省略部分输出信息………………
复制代码
6 S$ J& m! z5 p5 q2 o# I! X$ O4 @

: Q" h; s2 @$ m4 G2 d; B4.2.2 接收用户的参数但是,像上面这样的脚本程序只能执行一些预先定义好的功能,未免太过死板。为了让Shell脚本程序更好地满足用户的一些实时需求,以便灵活完成工作,必须要让脚本程序能够像之前执行命令时那样,接收用户输入的参数。; y9 l& G5 g1 e/ P( ?
比如,当用户执行某一个命令时,加或不加参数的输出结果是不同的:  j6 V. ?- a* T" p
  1. [root@linuxprobe ~]# wc -l anaconda-ks.cfg
    ! |: U. q* ?. d2 Y
  2. 44 anaconda-ks.cfg
    , g# c: x" Y6 V
  3. [root@linuxprobe ~]# wc -c anaconda-ks.cfg ! R  w) l1 l0 K3 I. v# Z8 A" x1 b
  4. 1407 anaconda-ks.cfg
    ; I- m' s1 k4 Z. ^! F
  5. [root@linuxprobe ~]# wc -w anaconda-ks.cfg 4 q1 ^) ]! K4 N: P! Q% m# a( G: V7 u* X4 i
  6. 121 anaconda-ks.cfg
复制代码
( V* F% s3 C9 y6 o1 x( Y" h2 i
" E- o& D. O) }) v: d
这意味着命令不仅要能接收用户输入的内容,还要有能力进行判断区别,根据不同的输入调用不同的功能。4 ?9 U" X% J' W  U
其实,Linux系统中的Shell脚本语言早就考虑到了这些,已经内设了用于接收参数的变量,变量之间使用空格间隔。例如,$0对应的是当前Shell脚本程序的名称,$#对应的是总共有几个参数,$*对应的是所有位置的参数值,$?对应的是显示上一次命令的执行返回值,而$1、$2、$3……则分别对应着第N个位置的参数值,如图4-15所示。% P' ~. Q3 R. m7 r
图4-15  Shell脚本程序中的参数位置变量
理论过后再来练习一下。尝试编写一个脚本程序示例,通过引用上面的变量参数来看一下真实效果:
* g/ W, P# t; y& N. m
  1. [root@linuxprobe ~]# vim example.sh
    + f& h) E6 b2 _! N- i
  2. #!/bin/bash# b3 X8 V7 ~  c1 G0 J. n
  3. echo "当前脚本名称为$0"
    7 L6 C) {1 ~/ n5 @
  4. echo "总共有$#个参数,分别是$*。"- C8 I, H/ i: y$ T6 ]& B
  5. echo "第1个参数为$1,第5个为$5。"7 L" O# ~' m: \- b8 Q; e
  6. [root@linuxprobe ~]# bash example.sh one two three four five six1 a8 L. a: _1 i5 @& B( g$ {
  7. 当前脚本名称为example.sh
    8 E3 u7 @! u+ m% q5 G/ b
  8. 总共有6个参数,分别是one two three four five six。5 v8 c. P7 l: m6 I/ A) v8 Q" Q
  9. 第1个参数为one,第5个为five。
复制代码
0 ?4 r$ V. u2 K2 \
8 K( t( n7 q. D+ X
4.2.3 判断用户的参数学习是一个登堂入室、由浅入深的过程。在学习完Linux命令,掌握Shell脚本语法变量和接收用户输入的信息之后,就要踏上新的高度—能够进一步处理接收到的用户参数。4 E7 t$ F: ~3 i4 Q2 L( u
本书在前面章节中讲到,系统在执行mkdir命令时会判断用户输入的信息,即判断用户指定的文件夹名称是否已经存在,如果存在则提示报错;反之则自动创建。Shell脚本中的条件测试语法可以判断表达式是否成立,若条件成立则返回数字0,否则便返回非零值。条件测试语法的执行格式如图4-16所示。切记,条件表达式两边均应有一个空格。+ |  u/ d' e3 F
图4-16  条件测试语句的执行格式

按照测试对象来划分,条件测试语句可以分为4种:2 B$ v  O! z* ]% U2 e5 C
文件测试语句;
, I: n& d% d6 D* I6 O5 \* b" p; G逻辑测试语句;# k+ M% I* ?8 P5 H% Z$ `: a" T! v
整数值比较语句;0 v3 u) _; g  P0 J, F6 ]8 \$ ~
字符串比较语句。* s  ^. Y; s0 V/ m: A5 g3 [, `
文件测试即使用指定条件来判断文件是否存在或权限是否满足等情况的运算符,具体的参数如表4-3所示。' a! ^3 U! G: |
表4-3                                                    文件测试所用的参数: M2 U4 @, n" f, N: u0 w9 E
操作符作用
-d测试文件是否为目录类型
-e测试文件是否存在
-f判断是否为一般文件
-r测试当前用户是否有权限读取
-w测试当前用户是否有权限写入
-x测试当前用户是否有权限执行
下面使用文件测试语句来判断/etc/fstab是否为一个目录类型的文件,然后通过Shell解释器的内设$?变量显示上一条命令执行后的返回值。如果返回值为0,则目录存在;如果返回值为非零的值,则意味着它不是目录,或这个目录不存在:
* q: {% L$ ~* z$ s4 A% n
( l. @# G- ^5 H% p% ]5 g# [9 o5 R
  1. [root@linuxprobe ~]# [ -d /etc/fstab ]$ z6 h6 t' z/ d# f' ^4 I9 b
  2. [root@linuxprobe ~]# echo $?
    3 F) l1 }5 f0 }# c
  3. 1" W! I; E: `1 p. k4 D& W1 x# G

  4. ( }8 ?+ w# v; i5 q
  5. 再使用文件测试语句来判断/etc/fstab是否为一般文件,如果返回值为0,则代表文件存在,且为一般文件:
    + B7 w. Q' g( A8 ]1 ~4 o6 O3 K  m
  6. ; y% \) f; w% ]! y5 g( m
  7. [root@linuxprobe ~]# [ -f /etc/fstab ]
    " w& {* C3 m) |5 X
  8. [root@linuxprobe ~]# echo $?2 x9 }* r& n; S9 E) D2 s1 I' o
  9. 0
    0 ]: G1 J/ L  \- f
复制代码

9 D: v) _9 T! k4 e0 i1 P: N# }; ^
+ C6 y" Q" v# Q. D' t* S8 h判断与查询一定要敲两次命令吗?其实可以一次搞定。
/ V1 E( U4 _; U! a( |4 Q. `逻辑语句用于对测试结果进行逻辑分析,根据测试结果可实现不同的效果。例如在Shell终端中逻辑“与”的运算符号是&&,它表示当前面的命令执行成功后才会执行它后面的命令,因此可以用来判断/dev/cdrom文件是否存在,若存在则输出Exist字样。( v; g3 {* P' D( z1 n( |
  1. [root@linuxprobe ~]# [ -e /dev/cdrom ] && echo "Exist"' Z( {! v2 r* D% H% j; Q- o: u
  2. Exist
复制代码
. i& R. q' r* q. M8 \* p
2 I& N, x+ |: q
除了逻辑“与”外,还有逻辑“或”,它在Linux系统中的运算符号为||,表示当前面的命令执行失败后才会执行它后面的命令,因此可以用来结合系统环境变量USER来判断当前登录的用户是否为非管理员身份:
' s. }. P: e  r: k3 b: G7 o
  1. [root@linuxprobe ~]# echo $USER
    4 L; v# w7 q, [
  2. root5 g4 u1 J' `/ M$ E2 g% K
  3. [root@linuxprobe ~]# [ $USER = root ] || echo "user"+ m: N" U, }2 l  d0 ~* e5 L
  4. [root@linuxprobe ~]# su - linuxprobe
    * }* l: l5 n$ S& j" Q3 d
  5. [linuxprobe@linuxprobe ~]$ [ $USER = root ] || echo "user"
    9 \1 \6 M! w0 |) I
  6. user
复制代码

1 C# w! T" [6 x( \. Q  z( X7 E' k9 _5 E/ c
第三种逻辑语句是“非”,在Linux系统中的运算符号是一个叹号(!),它表示把条件测试中的判断结果取相反值。也就是说,如果原本测试的结果是正确的,则将其变成错误的;原本测试错误的结果,则将其变成正确的。: O( y8 [+ g7 \, L5 W3 p
我们现在切换回到root管理员身份,再判断当前用户是否为一个非管理员的用户。由于判断结果因为两次否定而变成正确,因此会正常地输出预设信息:
, p7 p/ H4 C5 a5 ?& _/ Z
  1. [linuxprobe@linuxprobe ~]$ exit
    . Q; c* H% f+ t% Z. N
  2. logout
    $ P- M1 G6 q$ `
  3. [root@linuxprobe ~]# [ ! $USER = root ] || echo "administrator"4 E' P! N1 {# u) @* Y
  4. administrator
复制代码

' j. |% D8 H+ D" p" d3 C
8 a9 E6 X/ y- s3 G叹号应该放到判断语句的前面,代表对整个的测试语句进行取反值操作,而不应该写成“$USER != root”,因为“!=”代表的是不等于符号(≠),尽管执行效果一样,但缺少了逻辑关系,这一点还请多加注意。
! S" B- X# a- ]3 E. C0 S; eTips, O: f1 m, Z3 Y* ?/ D
&&是逻辑“与”,只有当前面的语句执行成功的时候才会执行后面的语句。
! J( V" `( |+ z" h8 {, D" n||是逻辑“或”,只有当前面的语句执行失败的时候才会执行后面的语句。
  R1 o! n4 x: [$ s- r) e!是逻辑“非”,代表对逻辑测试结果取反值;之前若为正确则变成错误,若为错误则变成正确。
* a$ t$ @" T- ]: [: ?: x% {* {! F
7 L2 i3 E% F+ P$ y  P3 i
* Q+ k2 I' x- m就技术图书的写作来讲,一般有两种套路:让读者真正搞懂技术了;让读者觉得自己搞懂技术了。因此市面上很多浅显的图书会让读者在学完之后感觉进步特别快,这基本上是作者有意为之,目的就是让您觉得“图书很有料,自己收获很大”,但是在步入工作岗位后就露出短板吃大亏。所以刘遄老师决定继续提高难度,为读者增加一个综合的示例,一方面作为前述知识的总结,另一方面帮助读者夯实基础,以便在今后的工作中更灵活地使用逻辑符号。
5 h& K# F# z+ g& A5 Z5 v当前我们正在登录的即为管理员用户—root。下面这个示例的执行顺序是,先判断当前登录用户的USER变量名称是否等于root,然后用逻辑“非”运算符进行取反操作,效果就变成了判断当前登录的用户是否为非管理员用户。最后若条件成立,则会根据逻辑“与”运算符输出user字样;若条件不满足,则会通过逻辑“或”运算符输出root字样,而只有在前面的&&不成立时才会执行后面的||符号。  k/ h5 `3 Y1 s% `- w( F: C5 Z
  1. [root@linuxprobe ~]# [ ! $USER = root ] && echo "user" || echo "root"9 i$ d& Y7 o0 G( X  O
  2. root
复制代码
! }: b8 m& A- d: J8 V9 d

- W  t/ s8 Q: m; [$ f5 A: d
整数比较运算符仅是对数字的操作,不能将数字与字符串、文件等内容一起操作,而且不能想当然地使用日常生活中的等号、大于号、小于号等来判断。因为等号与赋值命令符冲突,大于号和小于号分别与输出重定向命令符和输入重定向命令符冲突。因此一定要使用规范的整数比较运算符来进行操作。可用的整数比较运算符如表4-4所示。
! P5 e: F% S$ q表4-4                                                  可用的整数比较运算符8 O2 t  U; E& j  ~$ E3 C
操作符作用
-eq是否等于
-ne是否不等于
-gt是否大于
-lt是否小于
-le是否等于或小于
-ge是否大于或等于
接下来小试牛刀。先测试一下10是否大于10以及10是否等于10(通过输出的返回值内容来判断):
" U3 t" h: }- z' W7 A1 l0 ~( ^! `
  1. [root@linuxprobe ~]# [ 10 -gt 10 ]4 v; l0 Q1 v. U8 Y
  2. [root@linuxprobe ~]# echo $?4 H, L0 a6 j; D8 Z4 x
  3. 1
    8 G. C% c& I2 u9 `
  4. [root@linuxprobe ~]# [ 10 -eq 10 ]
    : O3 X3 V4 ^4 D: P, z- D0 N4 v
  5. [root@linuxprobe ~]# echo $?& ^8 X  G" n: h, g% k5 k9 H4 x
  6. 0
复制代码
# O8 k4 T: q$ W) n
; ]  b8 g! L; a  P/ F, y
在2.4节曾经讲过free命令,它能够用来获取当前系统正在使用及可用的内存量信息。接下来先使用free -m命令查看内存使用量情况(单位为MB),然后通过“grep Mem:”命令过滤出剩余内存量的行,再用awk '{print $4}'命令只保留第4列。这个演示确实有些难度,但看懂后会觉得很有意思,没准在运维工作中也会用得上。% ]" t5 \4 E. |$ C( G! M
  1. [root@linuxprobe ~]# free -m- C" J% V9 q: z3 E; }$ d; j
  2.               total        used        free      shared  buff/cache   available3 M3 o* [; v; X! d" F
  3. Mem:           1966        1374         128          16         463         397
    & A: q4 N( W2 y' D' n# M& z7 a
  4. Swap:          2047          66        1981
    & I- p& R% L4 I# k% V  D5 u% U
  5. [root@linuxprobe ~]# free -m | grep Mem:
    : G+ B9 K+ B7 Y4 o; |1 O5 @7 }
  6. Mem:           1966        1374         128          16         463         397
    + E/ F( D6 \" W7 m
  7. [root@linuxprobe ~]# free -m | grep Mem: | awk '{print $4}'
    ( |! E% \9 v' z; x& |! ]0 L
  8. 128
复制代码
' N, D* d5 ^% k# ~0 h* y; y

4 v8 R7 I2 N' l2 B如果想把这个命令写入到Shell脚本中,那么建议把输出结果赋值给一个变量,以方便其他命令进行调用:
/ q* q5 v) [  Y
  1. [root@linuxprobe ~]# FreeMem=`free -m | grep Mem: | awk '{print $4}'`7 R) ^" t- Y: G& k: z, D( W
  2. [root@linuxprobe ~]# echo $FreeMem
    # M# W, A1 t& {, v
  3. 128
复制代码
* @, Q) j! y; X/ ~; U2 d- U7 C9 `8 M; c
# G" q. W: y( C0 |' e. S+ R  e5 \
上面用于获取内存可用量的命令以及步骤可能有些“超纲”了,如果不能理解领会也不用担心,接下来才是重点。我们使用整数运算符来判断内存可用量的值是否小于1024,若小于则会提示“Insufficient Memory”(内存不足)的字样:+ }+ v" A9 R. H& D+ q1 A- x. ?6 v; O. P
  1. [root@linuxprobe ~]# [ $FreeMem -lt 1024 ] && echo "Insufficient Memory"
    + {5 Q8 |0 o" j* |% a
  2. Insufficient Memory
复制代码
  A1 H+ m6 o8 y+ e

8 l. ^/ Z& i! I4 Y. d  u7 h字符串比较语句用于判断测试字符串是否为空值,或两个字符串是否相同。它经常用来判断某个变量是否未被定义(即内容为空值),理解起来也比较简单。字符串比较中常见的运算符如表4-5所示。
- G. g# a0 s3 F# t, ^表4-5                                                常见的字符串比较运算符; T9 J/ J# {+ ^8 D1 K; n# j& K
操作符作用
=比较字符串内容是否相同
!=比较字符串内容是否不同
-z判断字符串内容是否为空
接下来通过判断String变量是否为空值,进而判断是否定义了这个变量:
& k6 D9 D/ A' @6 Y
  1. [root@linuxprobe ~]# [ -z $String ]
    . F5 g" P( S0 L& T
  2. [root@linuxprobe ~]# echo $?
    " u2 b& k! g; z) R' {
  3. 0
复制代码

  ^, h" u5 h, a" p- j
) V3 F% c# ~2 X8 s, E) O再次尝试引入逻辑运算符来试一下。当用于保存当前语系的环境变量值LANG不是英语(en.US)时,则会满足逻辑测试条件并输出“Not en.US”(非英语)的字样:" |9 n! Q' T/ s2 v8 t& x
  1. [root@linuxprobe ~]# echo $LANG7 A3 `# S' q1 m" }! A
  2. en_US.UTF-8
    " ^2 e% }$ f) c8 O9 m% b  g% K
  3. [root@linuxprobe ~]# [ ! $LANG = "en.US" ] && echo "Not en.US"
    7 k( D/ q- u: p) Q. W1 W6 R' [& ^# A
  4. Not en.US
复制代码
' l' a4 _" H5 d! b% E' q1 y( O
4.3 流程控制语句尽管此时可以通过使用Linux命令、管道符、重定向以及条件测试语句来编写最基本的Shell脚本,但是这种脚本并不适用于生产环境。原因是它不能根据真实的工作需求来调整具体的执行命令,也不能根据某些条件实现自动循环执行。通俗来讲,就是不能根据实际情况做出调整。( ]% ?6 J  r- Y, c5 j: d5 b' W
通常脚本都是从上到下一股脑儿地执行,效率是很高,但一旦某条命令执行失败了,则后面的功能全都会受到影响。假如大家有一天遇到了心仪的他(她),心中默默地进行如下规划(见图4-17)。' b; B) G$ a9 C
图4-17  心中规划
结果可能是见面聊天后就觉得不合适了,后续的“要手机号码”“一起吃晚饭”和“一起看电影”就要终止了,就需要转而去做其他事情,因此需要判断语句来帮助完成。" I! k; u1 A  M+ X$ [# I
接下来我们通过if、for、while、case这4种流程控制语句来学习编写难度更大、功能更强的Shell脚本。为了保证下文的实用性和趣味性,做到寓教于乐,我会尽可能多地讲解各种不同功能的Shell脚本示例,而不是逮住一个脚本不放,在它原有内容的基础上修修补补。尽管这种修补式的示例教学也可以让读者明白理论知识,但是却无法开放思路,不利于日后的工作。
7 [6 l' ^% n5 Q( j$ S: l. x4.3.1 if条件测试语句if条件测试语句可以让脚本根据实际情况自动执行相应的命令。从技术角度来讲,if语句分为单分支结构、双分支结构、多分支结构;其复杂度随着灵活度一起逐级上升。- T. G& b" e# n" G* u
if条件语句的单分支结构由if、then、fi关键词组成,而且只在条件成立后才执行预设的命令,相当于口语的“如果……那么……”。单分支的if语句属于最简单的一种条件判断结构,语法格式如图4-18所示。
$ F- R3 v" \3 \
图4-18  单分支的if条件语句
下面使用单分支的if条件语句来判断/media/cdrom目录是否存在,若不存在就创建这个目录,反之则结束条件判断和整个Shell脚本的执行。
( e. X0 ~" |% |) ?7 G2 ^* b9 ]
  1. [root@linuxprobe ~]# vim mkcdrom.sh
    ; M5 N7 r: x. x; P/ J: B, B) h
  2. #!/bin/bash7 a1 z+ g- c& s4 ?. L
  3. DIR="/media/cdrom"
    # Y" Z; Y6 \. u% {# v* I
  4. if [ ! -d $DIR ]4 ~6 q1 c1 P. {4 j8 S, P, j
  5. then   
    9 ]8 O9 ~2 U% G$ \
  6.         mkdir -p $DIR
    2 `1 x9 G5 d; H* G: J9 z
  7. fi
复制代码
1 B3 u# v+ T, e! i0 R5 [# o
: _9 k- K, \4 z) g
由于第5章才讲解用户身份与权限,因此这里继续用“bash脚本名称”的方式来执行脚本。在正常情况下,顺利执行完脚本文件后没有任何输出信息,但是可以使用ls命令验证/media/cdrom目录是否已经成功创建:
0 d) K0 L2 L; p1 w& n3 t8 h# e
  1. [root@linuxprobe ~]# bash mkcdrom.sh
    9 ^- [7 t- m4 k& ~( F- p
  2. [root@linuxprobe ~]# ls -ld /media/cdrom
    3 r7 g; z8 g8 H$ X+ K& K
  3. drwxr-xr-x. 2 root root 6 Oct 13 21:34 /media/cdrom
复制代码

' o0 [2 D; K2 w- O$ [4 W+ k8 C6 p. \. ^( L' E2 u

if条件语句的双分支结构由if、then、else、fi关键词组成,它进行一次条件匹配判断,如果与条件匹配,则去执行相应的预设命令;反之则去执行不匹配时的预设命令,相当于口语的“如果……那么……或者……那么……”。if条件语句的双分支结构也是一种很简单的判断结构,语法格式如图4-19所示。+ O- R; W0 {- S+ H) O& j# L
图4-19 双分支的if条件语句
下面使用双分支的if条件语句来验证某台主机是否在线,然后根据返回值的结果,要么显示主机在线信息,要么显示主机不在线信息。这里的脚本主要使用ping命令来测试与对方主机的网络连通性,而Linux系统中的ping命令不像Windows一样尝试4次就结束,因此为了避免用户等待时间过长,需要通过-c参数来规定尝试的次数,并使用-i参数定义每个数据包的发送间隔,以及使用-W参数定义等待超时时间。* |4 |6 b9 Z1 Y
  1. [root@linuxprobe ~]# vim chkhost.sh7 ~9 _2 ?* c. I- C8 A% u
  2. #!/bin/bash
    # m. u& ]2 m  j' I4 d  w* T
  3. ping -c 3 -i 0.2 -W 3 $1 &> /dev/null
    9 i$ k* x0 T( x6 l; ^0 ^
  4. if [ $? -eq 0 ]
    5 y; b4 D/ Y. ^* ~( K0 r
  5. then2 u$ J0 P$ x" t0 M% o4 g9 ^
  6.         echo "Host $1 is On-line."
    - K/ f3 M# c7 P  k
  7. else; G9 h% b) S$ s9 ]
  8.         echo "Host $1 is Off-line."
    7 g! C& z* `" @8 z5 r( J
  9. fi
复制代码
9 T3 z0 i0 `# {% Z' P+ n6 }4 s
8 M9 w5 d9 c7 ~1 B5 o- k2 ~
我们在4.2.3节中用过$?变量,作用是显示上一次命令的执行返回值。若前面的那条语句成功执行,则$?变量会显示数字0,反之则显示一个非零的数字(可能为1,也可能为2,取决于系统版本)。因此可以使用整数比较运算符来判断$?变量是否为0,从而获知那条语句的最终判断情况。这里的服务器IP地址为192.168.10.10,我们来验证一下脚本的效果:! F9 F/ H- d+ h5 {+ z
  1. [root@linuxprobe ~]# bash chkhost.sh 192.168.10.104 d5 |* Z* f. l' t* u/ n9 s
  2. Host 192.168.10.10 is On-line.. I" E/ T- g9 t3 |
  3. [root@linuxprobe ~]# bash chkhost.sh 192.168.10.207 {- a  Z: B4 z" i! b
  4. Host 192.168.10.20 is Off-line.
复制代码

5 g. S5 e) e* k# R: g5 ]
5 I& Q" z  t( V% }) vif条件语句的多分支结构由if、then、else、elif、fi关键词组成,它进行多次条件匹配判断,这多次判断中的任何一项在匹配成功后都会执行相应的预设命令,相当于口语的“如果……那么……如果……那么……”。if条件语句的多分支结构是工作中最常使用的一种条件判断结构,尽管相对复杂但是更加灵活,语法格式如图4-20所示。
* m4 g- F3 v- N8 t下面使用多分支的if条件语句来判断用户输入的分数在哪个成绩区间内,然后输出如Excellent、Pass、Fail等提示信息。在Linux系统中,read是用来读取用户输入信息的命令,能够把接收到的用户输入信息赋值给后面的指定变量,-p参数用于向用户显示一些提示信息。" x# Q! p, d+ d$ W. d/ `$ _% u
图 4-20 多分支的if条件语句
在下面的脚本示例中,只有当用户输入的分数大于等于85分且小于等于100分时,才输出Excellent字样;若分数不满足该条件(即匹配不成功),则继续判断分数是否大于等于70分且小于等于84分,如果是,则输出Pass字样;若两次都落空(即两次的匹配操作都失败了),则输出Fail字样:, C) w1 y# U: {1 f
  1. [root@linuxprobe ~]# vim chkscore.sh! J5 q0 u( N* x2 W2 k9 ~
  2. #!/bin/bash
    1 m4 G4 ?3 x! y0 Z
  3. read -p "Enter your score(0-100):" GRADE
    ) G: t: n7 N; B: m; {& e
  4. if [ $GRADE -ge 85 ] && [ $GRADE -le 100 ] ; then
    1 M, F6 ~& F( N. \1 @
  5.         echo "$GRADE is Excellent"
    0 A) N( U9 x$ [, f5 x6 l: d8 N
  6. elif [ $GRADE -ge 70 ] && [ $GRADE -le 84 ] ; then) G" \1 C8 |+ [: ~; d- b& p/ N4 v$ e
  7.         echo "$GRADE is Pass"
      z4 x* U+ u9 }
  8. else
    - x! [/ N8 ?0 d6 d% X
  9.         echo "$GRADE is Fail" 4 k8 q. d: n3 `3 w
  10. fi
    " S, S# i1 I* f. u4 c
  11. [root@linuxprobe ~]# bash chkscore.sh4 ?2 T. T/ |% w9 x4 c
  12. Enter your score(0-100):88
    9 X7 `' v! a4 e# V, t% s2 J% X
  13. 88 is Excellent
    ; j" u  k. c* b2 s0 v8 N
  14. [root@linuxprobe ~]# bash chkscore.sh / L7 u/ F. [) R8 U9 ?7 @
  15. Enter your score(0-100):80- X) Z" T; P& z% `% u4 @8 L
  16. 80 is Pass
复制代码
7 w5 f! O0 F, i4 M0 R% F

& r' j5 b1 H3 |- i! n下面执行该脚本。当用户输入的分数分别为30和200时,其结果如下:
) @! F' a7 `  k& c* l; u
  1. [root@linuxprobe ~]# bash chkscore.sh  
    , @. b1 n; z4 Q% r) p- O6 `! O
  2. Enter your score(0-100):30
    . I! T8 A! o; I$ o' }1 z# V
  3. 30 is Fail
    5 l5 ]6 ]% O3 H6 @5 a8 ]
  4. [root@linuxprobe ~]# bash chkscore.sh
    ; G. k' S# Z7 L8 f: p7 E$ k
  5. Enter your score(0-100):200 2 g$ Y- \0 u" g; W# l) @
  6. 200 is Fail
复制代码

# l% D1 ^$ X# o; O# v2 B( a  M
) I- u* _8 I5 ?9 I为什么输入的分数为200时,依然显示Fail呢?原因很简单—没有成功匹配脚本中的两个条件判断语句,因此自动执行了最终的兜底策略。可见,这个脚本还不是很完美,建议读者自行完善这个脚本,使得用户在输入大于100或小于0的分数时,给予Error报错字样的提示。5 s1 U/ J1 N% a! o% T0 [* s
4.3.2 for条件循环语句for循环语句允许脚本一次性读取多个信息,然后逐一对信息进行操作处理。当要处理的数据有范围时,使用for循环语句就再适合不过了。for循环语句的语法格式如图4-21所示。; z4 K2 r1 B9 y9 w# `0 w
图4-21  for范围循环语句
下面使用for循环语句从列表文件中读取多个用户名,然后为其逐一创建用户账户并设置密码。首先创建用户名称的列表文件users.txt,每个用户名称单独一行。读者可以自行决定具体的用户名称和个数:
) a& @- j6 V! W
  1. [root@linuxprobe ~]# vim users.txt2 {) |) C2 J8 z2 j
  2. andy/ m% g/ F* f  ?: K
  3. barry
    5 a- J! Y' Y: |: A8 |' S
  4. carl
    ' P" z" t. W: N$ U& t
  5. duke
    ( J! }  ~! v# @  A
  6. eric
    5 c9 r( b: F; s5 |9 A9 Y2 f
  7. george
复制代码
3 _- B* b. h4 M1 h# y: F

: s- B9 W8 w# I$ W9 [. e接下来编写Shell脚本addusers.sh。在脚本中使用read命令读取用户输入的密码值,然后赋值给PASSWD变量,并通过-p参数向用户显示一段提示信息,告诉用户正在输入的内容即将作为账户密码。在执行该脚本后,会自动使用从列表文件users.txt中获取到所有的用户名称,然后逐一使用“id用户名”命令查看用户的信息,并使用$?判断这条命令是否执行成功,也就是判断该用户是否已经存在。
( I) B* g+ R+ i5 ]! D2 O, h
  1. [root@linuxprobe ~]# vim addusers.sh
    1 N, j1 q9 Z& @- O) n
  2. #!/bin/bash
    - J! u5 V" W( W: j" o1 U! Y7 y3 y
  3. read -p "Enter The Users Password : " PASSWD8 w! ]1 e6 w; T+ E
  4. for UNAME in `cat users.txt`7 t" F) b# g9 c. G
  5. do
    / L; C  m* [1 P' k
  6.         id $UNAME &> /dev/null8 r& N0 y" Y6 O) |+ y
  7.         if [ $? -eq 0 ]
    ; ^$ I+ i& @4 q6 t! d1 ~
  8.         then. m# i' |! Z9 {7 X" c; T8 B
  9.                 echo "$UNAME , Already exists"
    1 @- i: x/ s- F
  10.         else
    2 {8 B  ~' G) Q# _
  11.                 useradd $UNAME5 e/ i3 J" @1 T, l% [- H
  12.                 echo "$PASSWD" | passwd --stdin $UNAME &> /dev/null6 ^1 }. w3 }8 @. Y. q6 a3 S+ y: |
  13.                 echo "$UNAME , Create success"
    6 R' ~% ?( g5 Y. T; B$ [! ]: o
  14.         fi
    + V& Z4 B  I8 n/ ^. o' ~& c; o, ]* O
  15. done
复制代码
  C  B* l7 n) z  v0 \
% r) ]4 p& c& |4 R  _
Tips9 c0 G1 g1 i7 O) N# e
/dev/null是一个被称作Linux黑洞的文件,把输出信息重定向到这个文件等同于删除数据(类似于没有回收功能的垃圾箱),可以让用户的屏幕窗口保持简洁。
3 r  T. \; _9 ~) d
& r/ }! B$ l* g执行批量创建用户的Shell脚本addusers.sh,在输入为账户设定的密码后将由脚本自动检查并创建这些账户。由于已经将多余的信息通过输出重定向符转移到了/dev/null黑洞文件中,因此在正常情况下屏幕窗口除了“用户账户创建成功”(Create success)的提示后不会有其他内容。
, P% b  |$ W0 M8 q; \在Linux系统中,/etc/passwd是用来保存用户账户信息的文件。如果想确认这个脚本是否成功创建了用户账户,可以打开这个文件,看其中是否有这些新创建的用户信息。
' U' P# `+ {: {5 R% M
  1. [root@linuxprobe ~]# bash addusers.sh# L! j! [  J7 Q' s/ q
  2. Enter The Users Password : linuxprobe
    # F. g9 }+ x+ A' T9 a" B  N1 B8 P
  3. andy , Create success2 ^, B4 d, P5 q% J# I
  4. barry , Create success
    3 I' ^6 u" R5 M1 [: L2 \) [0 m5 M: H
  5. carl , Create success( e8 A% e9 ^- v1 N8 w8 J4 L& W
  6. duke , Create success& ^; F- Q9 {: k
  7. eric , Create success( x) e! d) Z/ D7 W
  8. george , Create success7 b! ~9 x. W0 P( i
  9. [root@linuxprobe ~]# tail -6 /etc/passwd
    5 V+ M1 M9 H- z3 V6 @
  10. andy:x:1001:1001::/home/andy:/bin/bash' s% [% n5 x. ]9 t, m
  11. barry:x:1002:1002::/home/barry:/bin/bash& k6 a8 ?/ [& h; D6 `, ~; j
  12. carl:x:1003:1003::/home/carl:/bin/bash" `! R( k7 U* j7 p3 m$ i% M
  13. duke:x:1004:1004::/home/duke:/bin/bash, K' O( N* `# ]& n4 S
  14. eric:x:1005:1005::/home/eric:/bin/bash
    * a  i' ?7 ?' `
  15. george:x:1006:1006::/home/george:/bin/bash
复制代码
- m4 E0 A1 K! m7 U! [1 J$ ]$ M# ?

2 O- o) G. U% t# Z& Q大家还记得在学习双分支if条件语句时,用到的那个测试主机是否在线的脚本么?既然我们现在已经掌握了for循环语句,不妨做些更酷的事情,比如尝试让脚本从文本中自动读取主机列表,然后自动逐个测试这些主机是否在线。
& I, M4 S( X) Q& `5 F首先创建一个主机列表文件ipaddrs.txt:0 \3 Z4 z$ y- _5 x; L! [
  1. [root@linuxprobe ~]# vim ipaddrs.txt
    . l: d& a: |$ K5 y8 z# ]
  2. 192.168.10.100 [1 i. Q: B9 t: O' Z/ y, C
  3. 192.168.10.119 R5 {  Y5 y; v4 L% R
  4. 192.168.10.12
复制代码

( {- Q0 f* S4 U) ~
) i% K5 E% K: M5 e2 q  u! m/ B+ C然后将前面的双分支if条件语句与for循环语句相结合,让脚本从主机列表文件ipaddrs.txt中自动读取IP地址(用来表示主机)并将其赋值给HLIST变量,从而通过判断ping命令执行后的返回值来逐个测试主机是否在线。脚本中出现的“$(命令)”是一种完全类似于第3章的转义字符中反引号`命令`的Shell操作符,效果同样是执行括号或双引号括起来的字符串中的命令。大家在编写脚本时,多学习几种类似的新方法,可在工作中大显身手:
% i+ I2 t0 B$ w6 E% _! z+ C4 u
  1. [root@linuxprobe ~]# vim CheckHosts.sh8 K8 M  v6 K* H" o; U0 z9 D1 D* \2 w- l
  2. #!/bin/bash& ?9 ~: U. j1 k* j8 O/ |, k* O
  3. HLIST=$(cat ~/ipaddrs.txt)
    0 R: k! R# u/ f/ L7 E) W  S8 Z$ f
  4. for IP in $HLIST
    , |0 |' W9 u3 ^5 l8 `& p
  5. do
    & V+ ~+ r  S" l& A6 r6 N1 h
  6.         ping -c 3 -i 0.2 -W 3 $IP &> /dev/null
    0 O$ |3 o7 H. G$ k) t# E
  7.         if [ $? -eq 0 ]  
    6 C  u0 h5 V& ]
  8.         then
    ; ], q" N/ S* z2 P  e$ X
  9.                 echo "Host $IP is On-line."
    9 ]8 S% r  t& E/ s2 R0 n7 @' r: `
  10.         else- a  a" O( r2 S. Q: f& L
  11.                 echo "Host $IP is Off-line."5 V5 x' [9 B  k2 Q
  12.         fi
    * w9 D7 X8 X$ B3 l4 l6 o
  13. done
    . O- e2 }* g6 C* t  k
  14. [root@linuxprobe ~]# ./CheckHosts.sh
    5 s, r1 v. `! @, ^9 l- U+ J5 p
  15. Host 192.168.10.10 is On-line.2 {* ^# M2 P; e8 ]. B% g. u
  16. Host 192.168.10.11 is Off-line.4 f& o4 D; S! ~* k
  17. Host 192.168.10.12 is Off-line.
复制代码

: f, p6 {8 @# e$ l' B. c  U5 D$ }* B9 C2 t& f; L; ~
细心的读者应该发现了,Shell脚本中的代码缩进格式会根据不同的语句而改变。这是由Vim编辑器自动完成的,用户无须进行额外操作。但是,如果您使用的是RHEL 7以前的版本,则没有这个自动缩进功能,不过功能不受影响,只是会影响阅读体验而已。
6 l0 }' n6 H, n4.3.3 while条件循环语句

while条件循环语句是一种让脚本根据某些条件来重复执行命令的语句,它的循环结构往往在执行前并不确定最终执行的次数,完全不同于for循环语句中有目标、有范围的使用场景。while循环语句通过判断条件测试的真假来决定是否继续执行命令,若条件为真就继续执行,为假就结束循环。while语句的语法格式如图4-22所示。


$ |! l* q; @# P8 ]* Y/ M3 f1 n8 ^4 x* D( ^2 c8 T# b4 J
图4-22  while条件循环语句
接下来结合使用多分支的if条件测试语句与while条件循环语句,编写一个用来猜测数值大小的脚本Guess.sh。该脚本使用$RANDOM变量来调取出一个随机的数值(范围为0~32767),然后将这个随机数对1000进行取余操作,并使用expr命令取得其结果,再用这个数值与用户通过read命令输入的数值进行比较判断。这个判断语句分为3种情况,分别是判断用户输入的数值是等于、大于还是小于使用expr命令取得的数值。当前,现在这些内容不是重点,我们要关注的是while条件循环语句中的条件测试始终为true,因此判断语句会无限执行下去,直到用户输入的数值等于expr命令取得的数值后,才运行exit 0命令,终止脚本的执行。  l! J. T; q3 ~1 x: Y+ f
  1. [root@linuxprobe ~]# vim Guess.sh3 K% i$ W0 I- W5 S3 D8 }
  2. #!/bin/bash0 A2 R! _8 y! o" p/ @) o
  3. PRICE=$(expr $RANDOM % 1000)7 y; L6 v2 y/ J, {: e$ Z
  4. TIMES=0
    " k+ K* h" }1 m4 z- k
  5. echo "商品实际价格为0-999之间,猜猜看是多少?"1 K6 E7 d9 b/ t7 P: W2 D
  6. while true5 d1 n' r( U* T1 S& G' c+ R: \9 V1 h
  7. do
    / V& D* k$ {# n# h0 _
  8.         read -p "请输入您猜测的价格数目:" INT
    0 G1 K; R, G. u; b2 D8 ^
  9.         let TIMES++
    9 p) c' v+ G$ a3 ]
  10.         if [ $INT -eq $PRICE ] ; then; Y  y4 I) ~4 W% M" N0 ]
  11.                 echo "恭喜您答对了,实际价格是 $PRICE"% D, j. d9 q9 l; |- z
  12.                 echo "您总共猜测了 $TIMES 次"% p+ k& i1 v8 W& C( p/ E8 T
  13.                 exit2 l' Y1 M" u) a1 V
  14.         elif [ $INT -gt $PRICE ] ; then6 d+ e3 g" m% m! w! f
  15.                 echo "太高了!"8 l. y& n$ [0 L
  16.         else7 M! i. w9 r5 e2 U. [. n
  17.                 echo "太低了!". y( i9 w* E% |
  18.         fi4 Y  X& P2 d1 I
  19. done
复制代码
/ `3 O$ E, {! e$ [) ~3 Q7 p2 K9 I+ L
9 |4 }* }* L$ F+ c
在这个Guess.sh脚本中,我们添加了一些交互式的信息,从而使得用户与系统的互动性得以增强。而且每当循环到let TIMES++命令时都会让TIMES变量内的数值加1,用来统计循环总计执行了多少次。这可以让用户得知在总共猜测了多少次之后,才猜对价格。
; s$ Y, S: w0 h5 J9 E  O" A
  1. [root@linuxprobe ~]# bash Guess.sh
    & A) A( Q1 q9 I
  2. 商品实际价格为0-999之间,猜猜看是多少?% d- R0 q  M" q$ ]: p: g+ M
  3. 请输入您猜测的价格数目:500% R- P7 b" q6 D, l7 R
  4. 太低了!
      a* K2 P, @, [0 ^+ k$ }4 f
  5. 请输入您猜测的价格数目:800
    0 z2 ^" a8 u+ x* O, r, q& f
  6. 太高了!7 Q  {+ N, k- t) k5 N4 k
  7. 请输入您猜测的价格数目:650
    ; {6 c, S; t- U3 v1 ?6 p: \' t
  8. 太低了!
    . ~4 r" Q8 S3 e" Q* H
  9. 请输入您猜测的价格数目:720/ a& h& U: g9 s1 q
  10. 太高了!5 y. P3 ]; `5 g9 l
  11. 请输入您猜测的价格数目:6909 }) @1 M' w6 k! [' t5 E$ A
  12. 太低了!
    - r. N+ W' ]1 u$ ?  J
  13. 请输入您猜测的价格数目:700
    ) C1 F/ f: C1 F. F7 @- b# ?! }
  14. 太高了!+ o. V- r8 R0 s5 H5 b- V
  15. 请输入您猜测的价格数目:695% q+ O  O  e# f! j8 ?' U
  16. 太高了!! O" @- O, Z3 x6 _
  17. 请输入您猜测的价格数目:692! E$ x/ ]. a( C2 `/ f2 c6 ~( T
  18. 太高了!" n9 t: L% x$ G1 F( D
  19. 请输入您猜测的价格数目:691( L; k  o( l) T" z2 ~* y- U
  20. 恭喜您答对了,实际价格是 691
    $ F; G* i! B! L% f$ D
  21. 您总共猜测了 9 次
    : l0 i1 F9 O" l& x5 K+ u
  22. 8 G$ s  c1 Q1 I2 Q0 {9 s/ Z% f
复制代码
2 m& D0 G5 S3 k8 |8 k4 L7 E  D
当条件为true(真)的时候,while语句会一直循环下去,只有碰到exit才会结束,所以同学们一定要记得加上exit哦。4.3.4 case条件测试语句如果您之前学习过C语言,看到这一小节的标题肯定会会心一笑:“这不就是switch语句嘛!”是的,case条件测试语句和switch语句的功能非常相似!case语句是在多个范围内匹配数据,若匹配成功则执行相关命令并结束整个条件测试;如果数据不在所列出的范围内,则会去执行星号(*)中所定义的默认命令。case语句的语法结构如图4-23所示。
) d. ]0 D4 x8 I% l+ Z3 ]
图4-23  case条件测试语句
在前文介绍的Guess.sh脚本中有一个致命的弱点—只能接受数字!您可以尝试输入一个字母,会发现脚本立即就崩溃了。原因是字母无法与数字进行大小比较,例如,“a是否大于等于3”这样的命题是完全错误的。必须有一定的措施来判断用户输入的内容,当用户输入的内容不是数字时,脚本能予以提示,从而免于崩溃。
; N: h( C+ Q4 c通过在脚本中组合使用case条件测试语句和通配符(详见第3章),完全可以满足这里的需求。接下来我们编写脚本Checkkeys.sh,提示用户输入一个字符并将其赋值给变量KEY,然后根据变量KEY的值向用户显示其值是字母、数字还是其他字符。
" U4 S6 o, ]6 H$ [8 l8 ~, H3 C
  1. [root@linuxprobe ~]# vim Checkkeys.sh
    / ?4 W8 Y/ k1 H% G+ Q# y8 S8 G: Q5 N8 b
  2. #!/bin/bash
    ; ^* ?, |0 U: q3 d0 c
  3. read -p "请输入一个字符,并按Enter键确认:" KEY% E6 K# E; j- o
  4. case "$KEY" in$ a1 v( a9 _0 O* s9 R: H7 T: [
  5.         [a-z]|[A-Z]), p# i- W8 F7 s$ r) c
  6.                 echo "您输入的是 字母。"
    # I% K* m0 b3 r0 x; h2 V+ {
  7.                 ;;$ N' l" N: [5 I
  8.         [0-9])
    % n( b- w0 y0 u- |: ]% g# O
  9.                 echo "您输入的是 数字。"4 V! Q% `8 _9 p6 }6 s
  10.                 ;;/ s/ w0 {' k  _  j: L! U
  11.         *)- z5 U7 k7 l9 k
  12.                 echo "您输入的是 空格、功能键或其他控制字符。"
    . v4 S. S; m7 _1 k2 V2 s& ]* h
  13. esac
    ) z5 K8 r! m# D
  14. [root@linuxprobe ~]# bash Checkkeys.sh
    & [& \9 _- e) y0 D
  15. 请输入一个字符,并按Enter键确认:6
    + r) r' {( x6 p) \& n2 a7 J5 v2 G
  16. 您输入的是 数字。$ U9 {: P! n2 e# h8 @  z
  17. [root@linuxprobe ~]# bash Checkkeys.sh
    6 s  {# i/ f( g+ h0 k: c' v! S
  18. 请输入一个字符,并按Enter键确认:p; C1 \- R: m) N
  19. 您输入的是 字母。% M7 N8 P1 A1 F; c- r
  20. [root@linuxprobe ~]# bash Checkkeys.sh  M, `: Y5 b0 S) e/ s
  21. 请输入一个字符,并按Enter键确认:^[[15~
    3 m' v2 e1 E- I* H
  22. 您输入的是 空格、功能键或其他控制字符。
复制代码

/ p$ k- r  t) {2 b% \0 I
/ D: a2 s& r0 @9 D4.4 计划任务服务程序经验丰富的系统运维工程师可以使得Linux在无须人为介入的情况下,在指定的时间段自动启用或停止某些服务或命令,从而实现运维的自动化。尽管我们现在已经有了功能彪悍的脚本程序来执行一些批处理工作,但是,如果仍然需要在每天凌晨两点敲击键盘回车键来执行这个脚本程序,就太痛苦了(当然,也可以训练您的小猫在半夜按下回车键)。接下来,刘遄老师将向大家讲解如何设置服务器的计划任务服务,把周期性、规律性的工作交给系统自动完成。
: a+ x' n* S& j' G6 z0 H5 R/ Z计划任务分为一次性计划任务与长期性计划任务,大家可以按照如下方式理解。
, J, C* \% @6 ]
一次性计划任务:今晚23:30重启网站服务。: _% k# f  h/ S9 Y0 N# ~
长期性计划任务:每周一的凌晨3:25把/home/wwwroot目录打包备份为backup.tar.gz。) p: J( q4 G$ P4 u9 t( S, C) [
顾名思义,一次性计划任务只执行一次,一般用于临时的工作需求。可以用at命令实现这种功能,只需要写成“at时间”的形式就行。如果想要查看已设置好但还未执行的一次性计划任务,可以使用at -l命令;要想将其删除,可以使用“atrm任务序号”。at命令中的参数及其作用如表4-6所示。" O4 A: \/ E5 l# Q, F. Q
表4-6                                               at命令的参数及其作用! F5 I5 W5 q, [+ k
参数作用
-f指定包含命令的任务文件
-q指定新任务名称
-l显示待执行任务列表
-d删除指定待执行任务
-m任务执行后给用户发邮件
在使用at命令来设置一次性计划任务时,默认采用的是交互式方法。例如,使用下述命令将系统设置为在今晚23:30自动重启网站服务。
4 x9 l% }, l4 a/ y. W# \
  1. [root@linuxprobe ~]# at 23:30
    ) r+ V; C! \% q/ S2 e$ w7 i
  2. warning: commands will be executed using /bin/sh. `2 E) w* D! {. o. a( H5 b
  3. at> systemctl restart httpd( e  N& ^$ j3 H+ G5 G. r
  4. at> 此处请同时按下<Ctrl>+<d>键来结束编写计划任务
    " }: Q$ t$ J/ Z/ |* I- ]
  5. job 1 at Wed Oct 14 23:30:00 2020, E0 c) P* {5 h
  6. [root@linuxprobe ~]# at -l
    9 x* E0 D2 d; b+ l7 p0 v/ d
  7. 1 Wed Oct 14 23:30:00 2020 a root
复制代码

" e- ~; S" f, ~+ q: m9 D% ~/ }; n  e3 n
看到warning提醒信息不要慌,at命令只是在告诉我们接下来的任务将由sh解释器负责执行。这与此前学习的Bash解释器基本一致,不需要有额外的操作。
/ T5 p, x: d$ h: V8 u另外,如果大家想挑战一下难度更大但简捷性更高的方式,可以把前面学习的管道符(任意门)放到两条命令之间,让at命令接收前面echo命令的输出信息,以达到通过非交互式的方式创建计划一次性任务的目的。) W" D- Q9 N2 S9 S! S. T$ r8 a. q
  1. [root@linuxprobe ~]# echo "systemctl restart httpd" | at 23:30
    / b0 F" F" l/ K$ y$ a# v/ U
  2. warning: commands will be executed using /bin/sh* J0 y+ }1 c8 {/ P. t
  3. job 2 at Wed Oct 14 23:30:00 20207 A( h: f4 t8 x' K  O; M
  4. [root@linuxprobe ~]# at -l
    ( Y1 I! E3 g* R1 e
  5. 1 Wed Oct 14 23:30:00 2020 a root( o1 b. l% ~, U  J7 f9 }8 }% h0 M! O
  6. 2 Wed Oct 14 23:30:00 2020 a root
复制代码

( Y2 F0 F  {5 H8 R* }; V' F+ q$ p: O) n
上面设置了两条一样的计划任务,可以使用atrm命令轻松删除其中一条:
% ]: ~$ ~7 p0 E( y( N9 w5 L
  1. [root@linuxprobe ~]# atrm 29 R( b* |- ^8 m* r
  2. [root@linuxprobe ~]# at -l
    ) {  [4 G, h2 x* E- D
  3. 1 Wed Oct 14 23:30:00 2020 a root
复制代码

# g% ]( y2 J) T2 X5 L* ]
1 M/ U! b+ |% U" y9 U  ^1 R6 U% o这里还有一种特殊场景—把计划任务写入Shell脚本中,当用户激活该脚本后再开始倒计时执行,而不是像上面那样在固定的时间(“at 23:30”命令)进行。这该怎么办呢?0 h: v% W% n0 Y9 b
一般我们会使用“at now +2 MINUTE”的方式进行操作,这表示2分钟(MINUTE)后执行这个任务,也可以将其替代成小时(HOUR)、日(DAY)、月(MONTH)等词汇:2 g  @6 B* D# w6 c# m9 `
  1. [root@linuxprobe ~]# at now +2 MINUTE  y8 ~# M2 b8 m8 T
  2. warning: commands will be executed using /bin/sh' V. W6 g! P- n0 S, p4 a; e
  3. at> systemctl restart httpd
      E' L. E  `& G: O
  4. at> 此处请同时按下<Ctrl>+<d>键来结束编写计划任务
    2 O$ h0 W+ w  f3 C- E* L
  5. job 3 at Wed Oct 14 22:50:00 2020
复制代码
$ _5 n8 B! ], \; P+ f
7 U- `' x3 L8 w) z6 ^4 p

还有些时候,我们希望Linux系统能够周期性地、有规律地执行某些具体的任务,那么Linux系统中默认启用的crond服务简直再适合不过了。创建、编辑计划任务的命令为crontab -e,查看当前计划任务的命令为crontab -l,删除某条计划任务的命令为crontab -r。另外,如果您是以管理员的身份登录的系统,还可以在crontab命令中加上-u参数来编辑他人的计划任务。crontab命令中的参数及其作用如表4-7所示。0 Z- B" N2 X" X, \
表4-7                                               crontab命令的参数及其作用+ w6 O3 Y/ v4 W5 O$ n
参数作用
-e编辑计划任务
-u指定用户名称
-l列出任务列表
-r删除计划任务
在正式部署计划任务前,请先跟刘遄老师念一下口诀“分、时、日、月、星期 命令”。这是使用crond服务设置任务的参数格式(其格式见表4-8)。需要注意的是,如果有些字段没有被设置,则需要使用星号(*)占位,如图4-24所示。
; m6 ]+ g/ A7 V6 j
图4-24  使用crond设置任务的参数格式
表4-8                                       使用crond设置任务的参数字段说明) j: D( l, e* I) ~$ k2 G5 ^2 p+ s, q
字段说明
分钟取值为0~59的整数
小时取值为0~23的任意整数
日期取值为1~31的任意整数
月份取值为1~12的任意整数
星期取值为0~7的任意整数,其中0与7均为星期日
命令要执行的命令或程序脚本

* Q' u5 V' @5 s假设在每周一、三、五的凌晨3:25,都需要使用tar命令把某个网站的数据目录进行打包处理,使其作为一个备份文件。我们可以使用crontab -e命令来创建计划任务,为自己创建计划任务时无须使用-u参数。crontab –e命令的具体实现效果和crontab -l命令的结果如下所示:
* Z& P3 ^: W/ c  J
  1. [root@linuxprobe ~]# crontab -e' W. |8 ^& |" D7 t$ T& k0 E
  2. no crontab for root - using an empty one, r3 v7 ?, A4 C( E2 z- w0 h: t+ c
  3. crontab: installing new crontab
    2 R$ K1 q* K3 O6 \5 S$ d& w
  4. [root@linuxprobe ~]# crontab -l
    . g( c/ t; [9 K( s# _: `
  5. 25 3 * * 1,3,5 /usr/bin/tar -czvf backup.tar.gz /home/wwwroot
复制代码
1 |9 {( D; [4 b

+ ^9 A2 ?, F9 _" P4 f# C+ @0 n需要说明的是,除了用逗号(,)来分别表示多个时间段,例如“8,9,12”表示8月、9月和12月。还可以用减号(-)来表示一段连续的时间周期(例如字段“日”的取值为“12-15”,则表示每月的12~15日)。还可以用除号(/)表示执行任务的间隔时间(例如“*/2”表示每隔2分钟执行一次任务)。
  V* F3 L) _0 e; N/ C如果在crond服务中需要同时包含多条计划任务的命令语句,应每行仅写一条。例如我们再添加一条计划任务,它的功能是每周一至周五的凌晨1点自动清空/tmp目录内的所有文件。尤其需要注意的是,在crond服务的计划任务参数中,所有命令一定要用绝对路径的方式来写,如果不知道绝对路径,请用whereis命令进行查询。rm命令的路径为下面输出信息中的加粗部分。4 [" b' B3 U( s% m, M0 i
  1. [root@linuxprobe ~]# whereis rm" L! h3 z6 `7 E
  2. rm: /usr/bin/rm /usr/share/man/man1/rm.1.gz /usr/share/man/man1p/rm.1p.gz; W3 x  G6 L3 |4 l8 J) }
  3. [root@linuxprobe ~]# crontab -e6 L( E2 b) K5 d+ |
  4. crontab: installing new crontab
    - W/ {  ^. |9 T+ @4 C
  5. [root@linuxprobe ~]# crontab -l
    ; s% T  Y, z, {# K5 ?- Y
  6. 25 3 * * 1,3,5 /usr/bin/tar -czvf backup.tar.gz /home/wwwroot& n$ L2 v0 k0 d/ X" v
  7. 0  1 * * 1-5   /usr/bin/rm -rf /tmp/*
复制代码

' ~) J$ Y1 ^3 t  G: O* K1 ]  Y9 _* r* `- Y" b8 w/ n
总结一下使用计划服务的注意事项。  q6 b- [' _3 I  f$ z2 d
在crond服务的配置参数中,一般会像Shell脚本那样以#号开头写上注释信息,这样在日后回顾这段命令代码时可以快速了解其功能、需求以及编写人员等重要信息。% a3 Q. m" x" ]( d4 D: J
计划任务中的“分”字段必须有数值,绝对不能为空或是*号,而“日”和“星期”字段不能同时使用,否则就会发生冲突。, _; n+ @0 P1 b" d- j* q0 K
删除crond计划任务则非常简单,直接使用crontab -e命令进入编辑界面,删除里面的文本信息即可。也可以使用crontab -r命令直接进行删除:
: ]" V  C. f: k( n& L4 |5 ~9 B/ }
  1. [root@linuxprobe ~]# crontab -r( W6 C. q# i, q
  2. [root@linuxprobe ~]# crontab -l
    9 z$ p" y5 l6 t7 j- k
  3. no crontab for root
复制代码
) M7 w' K8 L9 k- y) H" Q
2 d8 t6 |0 S$ m% X; O# h, e# S1 ]% f
最后再啰唆一句,想必读者也已经发现了,诸如crond在内的很多服务默认调用的是Vim编辑器,相信大家现在能进一步体会到在Linux系统中掌握Vim文本编辑器的好处了吧。所以请大家一定要在彻底掌握Vim编码器之后再学习下一章。- q/ j6 e7 A8 \4 P1 A' q+ g: X

( D8 I# P, m2 L: B& s/ ~1 T/ N6 l
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

Archiver|手机版|小黑屋|返回首页

GMT+8, 2024-12-23 23:37 , Processed in 0.111606 second(s), 25 queries .

Powered by Discuz! X3.5

© 2001-2024 Discuz! Team.

快速回复 返回顶部 返回列表