第三节 Vim编辑器与Shell命令脚本
1.Vim文本编辑器
Vim英文全称为 Vi Improved,Vim 之所以能得到广大厂商与用户的认可,原因在于 Vim 编辑器中设置了 3 种模式—命令模式、末行模式和编辑模式,每种模式分别又支持多种不同的命令快捷键,这大大提高了工作效率,而且用户在习惯之后也会觉得相当顺手。要想高效地操作文本,就必须先搞清这 3 种模式的操作区别以及模式之间的切换方法。
➢ 命令模式:控制光标移动,可对文本进行复制、粘贴、删除和查找等工作。
➢ 输入模式:正常的文本录入。
➢ 末行模式:保存或退出文档,以及设置编辑环境。
在每次运行 Vim 编辑器时,默认进入命令模式,此时需要先切换到输入模式后再进行文档编写工作。而每次在编写完文档后需要先返回命令模式,然后再进入末行模式,执行文档的保存或退出操作。在 Vim 中,无法直接从输入模式切换到末行模式。Vim 编辑器中内置的命令有成百上千种用法,命令模式中最常用的一些命令如下:
末行模式主要用于保存或退出文件,以及设置 Vim 编辑器的工作环境,还可以让用户执行外部的 Linux 命令或跳转到所编写文档的特定行数。要想切换到末行模式,在命令模式中输入一个冒号就可以了。末行模式中常用的命令如下所示:
(1)编写简单文档
编写脚本文档的第 1 步就是给文档取个名字,这里将其命名为 practice.txt。如果存在该文档,则是打开它。如果不存在,则是创建一个临时的输入文件:
[root@localhost Desktop]# vim practice.txt
界面直接跳转到Vim编辑器的命令模式,。此时只能执行该模式下的命令,而不能随意输入文本内容。我们需要切换到输入模式才可以编写文档。可以分别使用 a、i、o 这 3 个键从命令模式切换到输入模式。其中,a键与 i 键分别是在光标后面一位和光标当前位置切换到输入模式,而 o 键则是在光标的下面再创建一个空行,此时可敲击 a 键进入编辑器的输入模式:
进入输入模式后,可以随意输入文本内容,Vim 编辑器不会把您输入的文本内容当作命令而执行:
在编写完之后,要想保存并退出,必须先敲击键盘的 Esc 键从输入模式返回命令模式,然后再输入“:wq!”切换到末行模式才能完成保存退出操作。(注:注意观察界面左下角的提示信息,在不同模式下有不同的提示字样)
然后可以使用cat命令查看保存后的文档内容:
[root@localhost ~]# cat practice.txt
Welcome to Vim World!
[root@localhost ~]#
继续编辑这个文档。因为要在原有文本内容的下面追加内容,所以在命令模式中敲击 o 键进入输入模式更会高效:
因为此时已经修改了文本内容,所以 Vim 编辑器在我们直接使用:q尝试直接退出文档而不保存的时候就会拒绝我们的操作了。此时只能强制退出:q!才能结束本次输入操作,在次使用cat命令查看能容,发现追加的内容没有保存。
(2)配置主机名称
在 Linux系统中,主机名大多保存在/etc/hostname 文件中,接下来将/etc/hostname 配置文件的内容修改为“linuxprobe.com”,步骤如下:
第一步:使用 Vim 编辑器修改/etc/hostname 主机名称文件。
第二步:把原始主机名称删除后追加“linuxprobe.com”。注意,使用 Vim 编辑器修改主机名称文件后,要在末行模式下执行“:wq!”命令才能保存并退出文档。
第三步:保存并退出文档,然后使用 hostname 命令检查是否修改成功
[root@localhost ~]# vim /etc/hostname
[root@localhost ~]# hostname
linuxprobe.com
(3)配置网卡信息
在 RHEL 5、RHEL 6 中,网卡配置文件的前缀为 eth,第 1 块网卡为eth0,第 2 块网卡为 eth1;以此类推。在 RHEL 7 中,网卡配置文件的前缀则以 ifcfg 开始,再加上网卡名称共同组成了网卡配置文件的名字,例如 ifcfg-eno16777736。而在 RHEL 8 中,网卡配置文件的前缀依然为 ifcfg,区别是网卡名称改成了类似于 ens160 的样子,不过好在除了文件名发生变化外,网卡参数没有其他大的区别。(注:本人系统安装的是RHEL 7)
现在有一个名称为 ifcfg-eno16777728 的网卡设备,将其配置为开机自启动,并且 IP 地址、子网、网关等信息由人工指定,其步骤如下所示。
第一步:首先切换到/etc/sysconfig/network-scripts 目录中(存放着网卡的配置文件)。
第二步:使用 Vim 编辑器修改网卡文件 ifcfg-eno16777728,逐项写入下面的配置参数并保存退
出。由于每台设备的硬件及架构是不一样的,因此请读者使用 ifconfig 命令自行确认各自网卡
的默认名称。
这是RHEL 8网卡信息:
➢ 设备类型:TYPE=Ethernet
➢ 地址分配模式:BOOTPROTO=static
➢ 网卡名称:NAME=eno16777728
➢ 是否启动:ONBOOT=yes
➢ IP地址:IPADDR=192.168.10.10
➢ 子网掩码:NETMASK=255.255.255.0
➢ 网关地址:GATEWAY=192.168.10.1
➢ DNS地址:DNS1=192.168.10.1
[root@localhost network-scripts]# vim ifcfg-eno16777728
[root@localhost network-scripts]# cat ifcfg-eno16777728
HWADDR=00:0C:29:3A:DC:A2
TYPE=Ethernet
BOOTPROTO=static
NAME=eno16777728
ONBOOT=yes
IPADDR=192.168.10.10
NETMASK=255.255.255.0
GATEWAY=192.168.10.1
DNS1=192.168.10.1
第三步:重启网络服务并测试网络是否连通。 (注:当前网络是不通的,所以显示结果没有问题)
[root@localhost network-scripts]# systemctl restart network
[root@localhost network-scripts]# ping 192.168.10.10
PING 192.168.10.10 (192.168.10.10) 56(84) bytes of data.
64 bytes from 192.168.10.10: icmp_seq=1 ttl=64 time=0.045 ms
64 bytes from 192.168.10.10: icmp_seq=2 ttl=64 time=0.035 ms
64 bytes from 192.168.10.10: icmp_seq=3 ttl=64 time=0.091 ms
64 bytes from 192.168.10.10: icmp_seq=4 ttl=64 time=0.058 ms
64 bytes from 192.168.10.10: icmp_seq=5 ttl=64 time=0.036 ms
64 bytes from 192.168.10.10: icmp_seq=6 ttl=64 time=0.036 ms
64 bytes from 192.168.10.10: icmp_seq=7 ttl=64 time=0.036 ms
64 bytes from 192.168.10.10: icmp_seq=8 ttl=64 time=0.035 ms
64 bytes from 192.168.10.10: icmp_seq=9 ttl=64 time=0.068 ms
64 bytes from 192.168.10.10: icmp_seq=10 ttl=64 time=0.037 ms
64 bytes from 192.168.10.10: icmp_seq=11 ttl=64 time=0.032 ms
64 bytes from 192.168.10.10: icmp_seq=12 ttl=64 time=0.037 ms
^C
--- 192.168.10.10 ping statistics ---
12 packets transmitted, 12 received, 0% packet loss, time 11000ms
rtt min/avg/max/mdev = 0.032/0.045/0.091/0.018 ms
(4)配置软件仓库
第一步:进入/etc/yum.repos.d/目录中(因为该目录存放着软件仓库的配置文件)。
第二步:使用 Vim 编辑器创建一个名为 rhel7.repo 的新配置文件(文件名称可随意,但后缀必须为.repo),逐项写入下面加粗的配置参数并保存退出(不要写后面的中文注释)。
➢ [rhel-media] :Yum 软件仓库唯一标识符,避免与其他仓库冲突。
➢ name=linuxprobe:Yum 软件仓库的名称描述,易于识别仓库用处。
➢ baseurl=file:///media/cdrom:提供的方式包括 FTP(ftp://..)、HTTP(http://..)、本地
(file:///..)。
➢ 是否启用(enabled =1) :设置此源是否可用;1 为可用,0 为禁用。
➢ 是否校验(gpgcheck=1):设置此源是否校验文件;1 为校验,0 为不校验。
➢ 公钥位置(gpgkey=file:///media/cdrom/RPM-GPG-KEY-redhat-release:):若上面的参数开启了校验功能,则此处为公钥文件位置。若没有开启,则省略不写。
第三步:按配置参数中所填写的仓库位置挂载光盘,并把光盘挂载信息写入/etc/fstab 文件中。
第四步:使用“yum install httpd -y”命令检查Yum软件仓库是否已经可用。
进入/etc/yum.repos.d 目录中后创建 Yum 配置文件:(暂不进行第3步和第4步,挂载出错,后面再进行补充)
[root@localhost yum.repos.d]# vim rhel7.repo
[root@localhost yum.repos.d]# cat rhel7.repo
[rhel7]
name=rhel7
baseurl=file:///media/cdrom
enabled=1
gpgcheck=0
2.编写Shell脚本
Shell 脚本命令的工作方式有下面两种:
➢ 交互式(Interactive):用户每输入一条命令就立即执行。
➢ 批处理(Batch):由用户事先编写好一个完整的 Shell 脚本,Shell 会一次性执行脚本
中诸多的命令。
通过查看 SHELL 变量可以发现,当前系统已经默认使用 Bash 作为命令行终端解释器了:
[root@localhost yum.repos.d]# echo $SHELL
/bin/bash
(1)编写简单的脚本
使用 Vim 编辑器把 Linux 命令按照顺序依次写入到一个文件中,就是一个简单的脚本了。
例如,如果想查看当前所在工作路径并列出当前目录下所有的文件及属性信息,实现这个功能的脚本应该类似于下面这样:
[root@localhost Desktop]# vim example.sh
[root@localhost Desktop]# cat example.sh
#!/bin/bash
#For Example BY linuxprobe.com
pwd
ls -al
注:Shell 脚本文件的名称可以任意,但为了避免被误以为是普通文件,建议将.sh 后缀加上,以表示是一个脚本文件。
在上面的这个 example.sh 脚本中实际上出现了 3 种不同的元素:
第一行的脚本声明(#!)用来告诉系统使用哪种 Shell 解释器来执行该脚本;
第二行的注释信息(#)是对脚本功能和某些命令的介绍信息,使得自己或他人在日后看到这个脚本内容时,可以快速知道该脚本的作用或一些警告信息;
第三、四行的可执行语句也就是我们平时执行的 Linux 命令了。
接下来执行一下这个文件看看结果:
[root@localhost Desktop]# bash example.sh
/root/Desktop
total 8
drwxr-xr-x. 2 root root 23 Nov 4 17:08 .
dr-xr-x---. 14 root root 4096 Nov 4 17:08 ..
-rw-r--r--. 1 root root 55 Nov 4 17:08 example.sh
[root@localhost Desktop]#
(2)接收用户的参数
Linux 系统中的 Shell 脚本语言早就考虑到了这些,已经内设了用于接收参数的变量,变量之间使用空格间隔。例如,$0 对应的是当前 Shell 脚本程序的名称,$#对应的是总共有几个参数,$*对应的是所有位置的参数值,$?对应的是显示上一次命令的执行返回值,而$1、$2、$3……则分别对应着第 N 个位置的参数值,如图 所示:
通过引用上面的变量参数来看一下真实效果:(当前系统不支持中文输入,一下显示的内容是先在本地编辑好之后,右键复制粘贴而来)
[root@localhost Desktop]# vim example.sh
[root@localhost Desktop]# cat example.sh
#!/bin/bash
#For Example BY linuxprobe.com
echo "当前脚本名称为$0。"
echo "总共有$#个参数,分别是$*。"
echo "第1个参数为$1,第5个参数为$5。"
执行命令后:
[root@localhost Desktop]# bash example.sh one two three four five six
当前脚本名称为example.sh。
总共有6个参数,分别是one two three four five six。
第1个参数为one,第5个参数为five。
(3)判断用户的参数
Shell 脚本中的条件测试语法可以判断表达式是否成立,若条件成立则返回数字 0,否则便返回非0值。条件测试语法的执行格式如图所示。切记,条件表达式两边均应有一个空格。
按照测试对象来划分,条件测试语句可以分为 4 种:
➢ 文件测试语句;
➢ 逻辑测试语句;
➢ 整数值比较语句;
➢ 字符串比较语句。
文件测试即使用指定条件来判断文件是否存在或权限是否满足等情况的运算符,具体的
参数如表所示:
下面使用文件测试语句来判断/etc/fstab 是否为一个目录类型的文件,然后通过 Shell 解释器的内设$?变量显示上一条命令执行后的返回值。如果返回值为 0,则目录存在;如果返回值为非零的值,则意味着它不是目录,或这个目录不存在:
[root@localhost Desktop]# [ -d /etc/fstab ]
[root@localhost Desktop]# echo $?
1
再使用文件测试语句来判断/etc/fstab 是否为一般文件,如果返回值为 0,则代表文件存在,且为一般文件:
[root@localhost Desktop]# [ -f /etc/fstab ]
[root@localhost Desktop]# echo $?
0
在 Shell终端中逻辑“与”的运算符号是&&,它表示当前面的命令执行成功后才会执行它后面的命令,因此可以用来判断/dev/cdrom 文件是否存在,若存在则输出 Exist 字样:
[root@localhost Desktop]# [ -e /dev/cdrom ] && echo "Exist"
Exist
除了逻辑“与”外,还有逻辑“或”,它在 Linux 系统中的运算符号为||,表示当前面的命令执行失败后才会执行它后面的命令,因此可以用来结合系统环境变量 USER 来判断当前登录的用户是否为非管理员身份:
[root@localhost Desktop]# echo $USER
root
[root@localhost Desktop]# [ $USER = root ] || echo "user"
[root@localhost Desktop]# su linuxprobe
[linuxprobe@linuxprobe Desktop]$ [ $USER = root ] || echo "user"
user
[linuxprobe@linuxprobe Desktop]$
第三种逻辑语句是“非”,在 Linux 系统中的运算符号是一个叹号(!),它表示把条件测试中的判断结果取相反值。也就是说,如果原本测试的结果是正确的,则将其变成错误的;原本测试错误的结果,则将其变成正确的。
我们现在切换回到 root 管理员身份,再判断当前用户是否为一个非管理员的用户。由于判断结果因为两次否定而变成正确,因此会正常地输出预设信息:
[linuxprobe@linuxprobe Desktop]$ exit
exit
[root@localhost Desktop]# [ ! $USER = root ] || echo "admin"
admin
[root@localhost Desktop]#
注:叹号应该放到判断语句的前面,代表对整个的测试语句进行取反值操作,而不应该写成“$USER != root”,因为“!=”代表的是不等于符号(≠),尽管执行效果一样,但缺少了逻辑关系,这一点还请多加注意。
总结:
➢ && 是逻辑“与”,只有当前面的语句执行成功的时候才会执行后面的语句。
➢ || 是逻辑“或”,只有当前面的语句执行失败的时候才会执行后面的语句。
➢ ! 是逻辑“非”,代表对逻辑测试结果取反值;之前若为正确则变成错误,若为错误则变成正确。
[root@localhost Desktop]# [ ! $USER = root ] && echo "user" || echo "root"
root
执行顺序:先判断当前登录用户的 USER 变量名称是否等于 root,然后用逻辑“非”运算符进行取反操作,效果就变成了判断当前登录的用户是否为非管理员用户。最后若条件成立,则会根据逻辑“与”运算符输出 user 字样;若条件不满足,则会通过逻辑“或”运算符输出 root 字样,而只有在前
面的&&不成立时才会执行后面的||符号。
注意事项:
整数比较运算符仅是对数字的操作,不能将数字与字符串、文件等内容一起操作,而且不能想当然地使用日常生活中的等号、大于号、小于号等来判断。因为等号与赋值命令符冲突,大于号和小于号分别与输出重定向命令符和输入重定向命令符冲突。因此一定要使用规范的整数比较运算符来进行操作。可用的整数比较运算符如表所示:
先测试一下 10 是否大于 10 以及 10 是否等于 10(通过输出的返回值内容来判断):
[root@localhost Desktop]# [ 10 -gt 10 ]
[root@localhost Desktop]# echo $?
1
[root@localhost Desktop]# [ 10 -eq 10 ]
[root@localhost Desktop]# echo $?
0
接下来先使用 free -m 命令查看内存使用量情况(单位为 MB),然后通过“grep Mem:”命令过滤出剩余内存量的行,再用 awk '{print $4}'命令只保留第 4 列:
[root@localhost Desktop]# free -m
total used free shared buffers cached
Mem: 1987 1544 443 9 0 387
-/+ buffers/cache: 1156 831
Swap: 2047 0 2047
[root@localhost Desktop]# free -m | grep Mem:
Mem: 1987 1542 445 9 0 387
[root@localhost Desktop]# free -m | grep Mem: | awk '{print $4}'
445
awk命令:可以让用户自定义函数或正则表达式,对文本内容进行高效管理,awk与sed、grep并称为Linux系统中的“文本三剑客”。
语法格式:awk 参数 文件名
如果想把这个命令写入到 Shell 脚本中,那么建议把输出结果赋值给一个变量,以方便其他命令进行调用:
[root@localhost Desktop]# FreeMem=`free -m | grep Mem: | awk '{print $4}'`
[root@localhost Desktop]# echo $FreeMem
395
注意:‘=’号左右不能有空格,否则报错
使用整数运算符来判断内存可用量的值是否小于 1024,若小于则会提示“Insufficient Memory”(内存不足)的字样:
[root@localhost Desktop]# [ $FreeMem -lt 1024 ] && echo "Insufficient Memory"
Insufficient Memory
字符串比较语句用于判断测试字符串是否为空值,或两个字符串是否相同。它经常用来判断某个变量是否未被定义(即内容为空值),理解起来也比较简单
来通过判断 String 变量是否为空值,进而判断是否定义了这个变量:
[root@localhost Desktop]# [ -z $String ]
[root@localhost Desktop]# echo $?
0
当用于保存当前语系的环境变量值 LANG 不是英语(en.US)时,则会满足逻辑测试条件并输出“Not en.US”(非英语)的字样:
[root@localhost Desktop]# echo $LANG
en_US.UTF-8
[root@localhost Desktop]# [ ! $LANG = "en.US" ] && echo "Not en.US"
Not en.US
3.流程控制语句
(1)if 条件测试语句
if 条件语句的单分支结构由 if、then、fi 关键词组成,而且只在条件成立后才执行预设的命令,相当于口语的“如果……那么……”。单分支的 if 语句属于最简单的一种条件判断结构,语法格式如图 所示。
使用单分支的 if 条件语句来判断/media/cdrom 目录是否存在,若不存在就创建这个目录,反之则结束条件判断和整个 Shell 脚本的执行。用“bash 脚本名称”的方式来执行脚本。在正常情况下,顺利执行完脚本文件后没有任何输出信息,但是可以使用 ls 命令验证/media/cdrom 目录是否已经成功创建:
[root@localhost Desktop]# vim mkcdrom.sh
[root@localhost Desktop]# cat mkcdrom.sh
#!/bin/bash
DIR="/media/cdrom"
if [ ! -d $DIR ]
then
mkdir -p $DIR
fi
[root@localhost Desktop]# bash mkcdrom.sh
[root@localhost Desktop]# ls -ld /media/cdrom
drwxr-xr-x. 2 root root 6 Nov 4 16:50 /media/cdrom
if 条件语句的双分支结构由 if、then、else、fi 关键词组成,它进行一次条件匹配判断,如果与条件匹配,则去执行相应的预设命令;反之则去执行不匹配时的预设命令,相当于口语的“如果……那么……或者……那么……”。if 条件语句的双分支结构也是一种很简单的判断结构,语法格式如图 所示。
下面使用双分支的 if 条件语句来验证某台主机是否在线,然后根据返回值的结果,要么显示主机在线信息,要么显示主机不在线信息。这里的脚本主要使用 ping 命令来测试与对方主机的网络连通性,而 Linux 系统中的 ping 命令不像 Windows 一样尝试 4 次就结束,因此为了避免用户等待时间过长,需要通过-c 参数来规定尝试的次数,并使用-i 参数定义每个数据包的发送间隔,以及使用-W 参数定义等待超时时间。
[root@localhost Desktop]# vim chkhost.sh
[root@localhost Desktop]# cat chkhost.sh
#!/bin/bash
ping -c 3 -i 0.2 -W 3 $1 &> /dev/null
if [ $? -eq 0 ]
then
echo "Host $1 is On-line."
else
echo "Host $1 if Off-line."
fi
[root@localhost Desktop]# bash chkhost.sh 192.168.10.11
Host 192.168.10.11 if Off-line.
if 条件语句的多分支结构由 if、then、else、elif、fi 关键词组成,它进行多次条件匹配判
断,这多次判断中的任何一项在匹配成功后都会执行相应的预设命令,相当于口语的“如
果……那么……如果……那么……”。if 条件语句的多分支结构是工作中最常使用的一种条件
判断结构,尽管相对复杂但是更加灵活,语法格式如图所示。
在下面的脚本示例中,只有当用户输入的分数大于等于 85 分且小于等于 100 分时,才输出 Excellent 字样;若分数不满足该条件(即匹配不成功),则继续判断分数是否大于等于 70分且小于等于 84 分,如果是,则输出 Pass 字样;若两次都落空(即两次的匹配操作都失败了),则输出 Fail 字样:(在 Linux 系统中,read 是用来读取用户输入信息的命令,能够把接收到的用户输入信息赋值给后面的指定变量,-p 参数用于向用户显示一些提示信息)
[root@localhost Desktop]# vim chkscore.sh
[root@localhost Desktop]# cat chkscore.sh
#!/bin/bash
read -p "Enter your score (0-100):" GRADE
if [ $GRADE -ge 85 ] && [ $GRADE -le 100 ] ; then
echo "$GRADE is Excellent"
elif [ $GRADE -ge 70 ] && [ $GRADE -le 84 ] ; then
echo "$GRADE is Pass"
else
echo "$GRADE is Fail"
fi
[root@localhost Desktop]# bash chkscore.sh
Enter your score (0-100):87
87 is Excellent
[root@localhost Desktop]# bash chkscore.sh
Enter your score (0-100):84
84 is Pass
[root@localhost Desktop]# bash chkscore.sh
Enter your score (0-100):65
65 is Fail
(2)for条件循环语句
for 循环语句允许脚本一次性读取多个信息,然后逐一对信息进行操作处理。当要处理的数据有范围时,使用 for 循环语句就再适合不过了。for 循环语句的语法格式如图所示。
使用 for 循环语句从列表文件中读取多个用户名,然后为其逐一创建用户账户并设置密码。首先创建用户名称的列表文件 users.txt,每个用户名称单独一行。
[root@localhost Desktop]# vim users.txt
[root@localhost Desktop]# cat users.txt
andy
barry
carl
接下来编写 Shell 脚本 addusers.sh。在脚本中使用 read 命令读取用户输入的密码值,然后赋值给 PASSWD 变量,并通过-p 参数向用户显示一段提示信息,告诉用户正在输入的内容即将作为账户密码。在执行该脚本后,会自动使用从列表文件 users.txt 中获取到所有的用户名称,然后逐一使用“id 用户名”命令查看用户的信息,并使用$?判断这条命令是否执行成功,也就是判断该用户是否已经存在。
[root@localhost Desktop]# vim users.txt
[root@localhost Desktop]# cat users.txt
andy
barry
carl
[root@localhost Desktop]# vim addusers.sh
[root@localhost Desktop]# cat addusers.sh
#!/bin/bash
read -p "Enter The Users Password:" PASSWD
for UNAME in `cat users.txt`
do
id $UNAME &> /dev/null
if [ $? -eq 0 ]
then
echo "$UNAME, Already exists."
else
useradd $UNAME &> /dev/null
echo "$PASSWD" | passwd --stdin $UNAME &> /dev/null
echo "$UNAME, Create success"
fi
done
[root@localhost Desktop]# bash addusers.sh
Enter The Users Password:123456
andy, Create success
barry, Create success
carl, Create success
[root@localhost Desktop]# tail -n 3 /etc/passwd
andy:x:1001:1001::/home/andy:/bin/bash
barry:x:1002:1002::/home/barry:/bin/bash
carl:x:1003:1003::/home/carl:/bin/bash
注:/dev/null 是一个被称作 Linux 黑洞的文件,把输出信息重定向到这个文件等同于删除数据(类似于没有回收功能的垃圾箱),可以让用户的屏幕窗口保持简洁。
(3)while条件循环语句
while 条件循环语句是一种让脚本根据某些条件来重复执行命令的语句,它的循环结构往往在执行前并不确定最终执行的次数,完全不同于 for 循环语句中有目标、有范围的使用场景。while 循环语句通过判断条件测试的真假来决定是否继续执行命令,若条件为真就继续执行,为假就结束循环。while 语句的语法格式如图所示。
编写一个用来猜测数值大小的脚本 Guess.sh。该脚本使用$RANDOM 变量来调取出一个随机的数值(范围为 0~32767),然后将这个随机数对 1000 进行取余操作,并使用 expr 命令取得其结果,再用这个数值与用户通过 read 命令输入的数值进行比较判断。这个判断语句分为 3 种情况,分别是判断用户输入的数值是等于、大于还是小于使用 expr 命令取得的数值。我们要关注的是 while 条件循环语句中的条件测试始终为 true,因此判断语句会无限执行下去,直到用户输入的数值等于 expr 命令取得的数值后,才运行 exit 0 命令,终止脚本的执行。
[root@localhost Desktop]# vim guess.sh
[root@localhost Desktop]# cat guess.sh
#!/bin/bash
PRICE=$(expr $RANDOM % 1000)
TIMES=0
echo "the real price is in 0-999,please try it?"
while true
do
read -p "please input your number: " INT
let TIMES++
if [ $INT -eq $PRICE ] ; then
echo "you are right, the real price is $PRICE"
echo "you guess $TIMES"
exit
elif [ $INT -gt $PRICE ] ; then
echo "too high"
else
echo "too low"
fi
done
[root@localhost Desktop]# bash guess.sh
the real price is in 0-999,please try it?
please input your number: 500
too high
please input your number: 200
too low
please input your number: 300
too high
please input your number: 250
too high
please input your number: 240
too high
please input your number: 210
too high
please input your number: 209
too high
please input your number: 208
too high
please input your number: 207
too high
please input your number: 206
too high
please input your number: 205
too high
please input your number: 204
you are right, the real price is 204
you guess 12
(4)case条件测试语句
case 语句是在多个范围内匹配数据,若匹配成功则执行相关命令并结束整个条件测试;如果数据不在所列出的范围内,则会去执行星号(*)中所定义的默认命令。case 语句的语法结构如图所示。
接下来我们编写脚本 Checkkeys.sh,提示用户输入一个字符并将其赋值给变量 KEY,然后根据变量 KEY 的值向用户显示其值是字母、数字还是其他字符。
[root@localhost Desktop]# vim checkkey.sh
[root@localhost Desktop]# cat checkkey.sh
#!/bin/bash
read -p "please input a char, then push enter:" KEY
case "$KEY" in
[a-z] | [A-Z])
echo "you input is alpha."
;;
[0-9])
echo "you input is number."
;;
*)
echo "you input is space,tab or others."
esac
[root@localhost Desktop]# bash checkkey.sh
please input a char, then push enter:7
you input is number.
[root@localhost Desktop]# bash checkkey.sh
please input a char, then push enter:o
you input is alpha.
[root@localhost Desktop]# bash checkkey.sh
please input a char, then push enter:
you input is space,tab or others.
[root@localhost Desktop]# bash checkkey.sh
please input a char, then push enter:'
you input is space,tab or others.
[root@localhost Desktop]# bash checkkey.sh
please input a char, then push enter:[
you input is space,tab or others.
4.计划任务服务程序
计划任务分为一次性计划任务与长期性计划任务。顾名思义,一次性计划任务只执行一次,一般用于临时的工作需求。可以用 at 命令实现这种功能,只需要写成“at 时间”的形式就行。如果想要查看已设置好但还未执行的一次性计划任务,可以使用 at -l 命令;要想将其删除,可以使用“atrm 任务序号”。at 命令中的参数及其作用如表所示。
在使用 at 命令来设置一次性计划任务时,默认采用的是交互式方法。例如,使用下述命令将系统设置为在今晚 23:30 自动重启网站服务。
[root@localhost Desktop]# at 23:30
at> systemctl restart httpd
at> <EOT>(此处按下ctl+D结束键盘输入)
job 1 at Tue Nov 5 23:30:00 2024
[root@localhost Desktop]# at -l
1Tue Nov 5 23:30:00 2024 a root
或者使用管道方式,让 at 命令接收前面 echo 命令的输出信息,以达到通过非交互式的方式创建计划一次性任务的目的。
[root@localhost Desktop]# echo "systemctl restart httpd" | at 24:00
job 2 at Wed Nov 6 00:00:00 2024
[root@localhost Desktop]# at -l
1Tue Nov 5 23:30:00 2024 a root
2Wed Nov 6 00:00:00 2024 a root
上面设置了两条计划任务,可以使用 atrm 命令轻松删除其中一条:
[root@localhost Desktop]# atrm 2
[root@localhost Desktop]# at -l
1Tue Nov 5 23:30:00 2024 a root
这里还有一种特殊场景—把计划任务写入 Shell 脚本中,当用户激活该脚本后再开始倒计时执行,而不是像上面那样在固定的时间(“at 23:30”命令)进行。
一般我们会使用“at now +2 MINUTE”的方式进行操作,这表示 2 分钟(MINUTE)后执行这个任务,也可以将其替代成小时(HOUR)、日(DAY)、月(MONTH)等词汇:
[root@localhost Desktop]# at now +2 MINUTE
at> systemctl restart httpd
at> <EOT>
job 3 at Tue Nov 5 16:52:00 2024
还有些时候,我们希望 Linux 系统能够周期性地、有规律地执行某些具体的任务,那么Linux 系统中默认启用的 crond 服务简直再适合不过了。创建、编辑计划任务的命令为 crontab -e,查看当前计划任务的命令为 crontab -l,删除某条计划任务的命令为 crontab -r。另外,如果您是以管理员的身份登录的系统,还可以在 crontab 命令中加上-u 参数来编辑他人的计划任务。crontab 命令中的参数及其作用如表所示。
在正式部署计划任务前,先记一下口诀“分、时、日、月、星期 命令”。这是使用 crond 服务设置任务的参数格式。需要注意的是,如果有些字段没有被设置,则需要使用星号(*)占位,如图所示。
假设在每周一、三、五的凌晨 3:25,都需要使用 tar 命令把某个网站的数据目录进行打包处理,使其作为一个备份文件。我们可以使用 crontab -e 命令来创建计划任务,为自己创建计划任务时无须使用-u 参数。crontab –e 命令的具体实现效果和 crontab -l 命令的结果如下所示:
[root@localhost ~]# crontab -e
no crontab for root - using an empty one
crontab: installing new crontab
[root@localhost ~]# crontab -l
25 3 * * 1,3,5 /usr/bin/tar -czvf backup.tar.gz /home/wwwroot
除了用逗号(,)来分别表示多个时间段,例如“8,9,12”表示 8 月、9 月和 12 月。还可以用减号(-)来表示一段连续的时间周期(例如字段“日”的取值为“12-15”,则表示每月的 12~15 日)。还可以用除号(/)表示执行任务的间隔时间(例如“*/2”表示每隔 2 分钟执行一次任务)。
如果在 crond 服务中需要同时包含多条计划任务的命令语句,应每行仅写一条。例如我们再添加一条计划任务,它的功能是每周一至周五的凌晨 1 点自动清空/tmp 目录内的所有文件。尤其需要注意的是,在 crond 服务的计划任务参数中,所有命令一定要用绝对路径的方式来写,如果不知道绝对路径,请用 whereis 命令进行查询。rm 命令的路径为下面输出信息中的加粗部分。
[root@localhost ~]# whereis rm
rm: /usr/bin/rm /usr/share/man/man1/rm.1.gz /usr/share/man/man1p/rm.1p.gz
[root@localhost ~]# crontab -e
crontab: installing new crontab
[root@localhost ~]# crontab -l
25 3 * * 1,3,5 /usr/bin/tar -czvf backup.tar.gz /home/wwwroot
0 1 * * 1-5 /usr/bin/rm -rf /tmp/*
总结一下使用计划服务的注意事项。
➢ 在 crond 服务的配置参数中,一般会像 Shell 脚本那样以#号开头写上注释信息,这样在日后回顾这段命令代码时可以快速了解其功能、需求以及编写人员等重要信息。
➢ 计划任务中的“分”字段必须有数值,绝对不能为空或是*号,而“日”和“星期”字段不能同时使用,否则就会发生冲突。
原文地址:https://blog.csdn.net/u012616827/article/details/143485732
免责声明:本站文章内容转载自网络资源,如本站内容侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!