计划任务 Shell 进阶(一)

1.并发运行
同时运行多个进程,提高效率
方法1
vi all.sh
脚本内容如下:
f1.sh&
f2.sh&
f3.sh&
方法2
(f1.sh&);(f2.sh&);(f3.sh&)
方法3
{ f1.sh& f2.sh& f3.sh& }

2.任务计划
Linux任务计划、周期性任务执行
•未来的某时间点执行一次任务
at
batch:系统自行选择空闲时间去执行此处指定的任务
•周期性运行某任务
cron

3.at任务
包:at
at命令:at [option] TIME
常用选项:
-V 显示版本信息:
-l: 列出指定队列中等待运行的作业;相当于atq
-d: 删除指定的作业;相当于atrm
-c: 查看具体作业任务
-f /path/from/somefile:从指定的文件中读取任务
-m:当任务被完成之后,将给用户发送邮件,即使没有标准输出
注意:作业执行命令的结果中的标准输出和错误以邮件通知给相关用户,不支持在屏幕上直接标准输出。
TIME:定义出什么时候进行 at 这项任务的时间
HH:MM [YYYY-mm-dd]
noon, midnight, teatime(4pm)
tomorrow
now+#{minutes,hours,days, OR weeks}

4.at时间格式
HH:MM 02:00
在今日的 HH:MM 进行,若该时刻已过,则明天此时执行任务
HH:MM YYYY-MM-DD 02:00 2016-09-20
规定在某年某月的某一天的特殊时刻进行该项任务
HH:MM[am|pm] [Month] [Date]
04pm March 17
17:20 tomorrow
HH:MM[am|pm] + number [minutes|hours|days|weeks]
在某个时间点再加几个时间后才进行该项任务
now + 5 minutes
02pm + 3 days

at任务
执行方式:
1)交互式 2)输入重定向 3)at –f 文件
依赖与atd服务,需要启动才能实现at任务
at队列存放在/var/spool/at目录中
/etc/at.{allow,deny}控制用户是否能执行at任务
白名单:/etc/at.allow 默认不存在,只有该文件中的用户才能执行at命令
黑名单:/etc/at.deny 默认存在,拒绝该文件中用户执行at命令,而没有在at.deny
文件中的使用者则可执行
白名单比黑名单优先级高,如果两个文件都有XX用户,优先白名单
如果两文件都不存在,所有user都不可以执行at,只有root可以执行 at 命令

5.周期性任务计划cron
周期性任务计划:cron
相关的程序包:
cronie: 主程序包,提供crond守护进程及相关辅助工具
cronie-anacron:cronie的补充程序,用于监控cronie任务执行状况,如cronie中的任务在过去该运行的时间点未能正常运行,则anacron会随后启动一次此任务
crontabs:包含CentOS提供系统维护任务
查看是否安装了包 rpm -ql cronie

计划任务
确保crond守护处于运行状态:
CentOS 7:
systemctl status crond
CentOS 6:
service crond status
计划周期性执行的任务提交给crond,到指定时间会自动运行
系统cron任务:系统维护作业
/etc/crontab
用户cron任务:
crontab命令
日志:/var/log/cron

系统cron任务:/etc/crontab
注释行以 # 开头
详情参见 man 5 crontab

Example of job definition:

.—————- minute (0 - 59)

| .————- hour (0 - 23)

| | .———- day of month (1 - 31)

| | | .——- month (1 - 12) OR jan,feb,mar,apr …

| | | | .—- day of week (0 - 6) (Sunday=0 or 7) OR sun,mon,tue,wed,thu,fri,sat

| | | | |

* * * * * user-name command to be executed

例如:晚上9点10分运行echo命令
10 21 * * * centos /bin/echo “Howdy!”

时间语法只能定义 或的关系,不支持与的关系,因为表示与的时间关系可以在被调用脚本里做时间判断。

时间表示法:
•(1) 特定值
给定时间点有效取值范围内的值
•(2) *
给定时间点上有效取值范围内的所有值
表示“每…”
•(3) 离散取值

,#,

•(4) 连续取值“-”横杠

-

•(5) 在指定时间范围上,定义步长“斜杠”
/#: #即为步长

(6) @符号的用法
@reboot Run once after reboot
@yearly 0 0 1 1 *
@annually 0 0 1 1 *
@monthly 0 0 1 * *
@weekly 0 0 * * 0
@daily 0 0 * * *
@hourly 0 * * * *
示例:每3小时echo和wall命令
0 /3 * * centos /bin/echo “howdy”;/usr/bin/wall “welcome to Magedu!”

@reboot root reboot 将进入死循环,重启完再重启,解决办法:
开启的时候按 E 键

输入rd.break 按Ctrl-x进入紧急救援模式

挂根并可写(默认只读,需要加-o)

系统的计划任务:
/etc/crontab
/etc/cron.d/ 配置文件
/etc/cron.hourly/ 脚本
/etc/cron.daily/ 脚本
/etc/cron.weekly/ 脚本
/etc/cron.monthly/ 脚本

6.anacron系统
运行计算机关机时cron不运行的任务,CentOS6以后版本取消anacron服务,由crond服务管理
对笔记本电脑、台式机、工作站、偶尔要关机的服务器及其它不一直开机的系统很重要对很有用
配置文件:/etc/anacrontab,负责执行/etc/ cron.daily /etc/cron.weekly /etc/cron.monthly中系统任务。
•字段1:如果在这些日子里没有运行这些任务……
•字段2:在重新引导后等待这么多分钟后运行它
•字段3:任务识别器,在日志文件中标识
•字段4:要执行的任务
由/etc/cron.hourly/0anacron执行
当执行任务时,更新/var/spool/anacron/cron.daily 文件的时间戳

7.管理临时文件
CentOS6使用/etc/cron.daily/tmpwatch定时清除临时文件
CentOS7使用systemd-tmpfiles-setup服务实现
配置文件:
/etc/tmpfiles.d/*.conf
/run/tmpfiles.d/*.conf
/usr/lib/tmpfiles/*.conf
/usr/lib/tmpfiles.d/tmp.conf
d /tmp 1777 root root 10d
d /var/tmp 1777 root root 30d
命令:
systemd-tmpfiles –clean|remove|create configfile
管理临时文件

8.用户计划任务
crontab命令定义
每个用户都有专用的cron任务文件: /var/spool/cron/USERNAME
crontab命令:
crontab [-u user] [-l | -r | -e] [-i]
-l: 列出所有任务
-e: 编辑任务
-r: 移除所有任务
-i:同-r一同使用,以交互式模式移除指定任务
-u user: 仅root可运行,指定用户管理cron任务
控制用户执行计划任务:
/etc/cron.{allow,deny}
黑白名单文件只能拒绝加入黑白名单操作之后的执行,但是之前的执行依然有效。
白名单比黑名单优先级高,如果两个文件都有XX用户,优先白名单
如果两文件都不存在,所有user都不可以执行crontab,只有root可以执行 crontab

9.at和crontab
一次性作业使用 at
重复性作业使用crontab

没有被重定向的输出会被邮寄给用户
根用户能够修改其它用户的作业

10.计划任务
注意:运行结果的标准输出和错误以邮件通知给相关用户
(1) COMMAND > /dev/null
(2) COMMAND &> /dev/null
对于cron任务来讲,%有特殊用途;如果在命令中要使用%,则需要转义,将%放置于单引号中,则可不用转义

思考:
(1) 如何在秒级别运行任务?
* * * * * for min in 0 1 2; do echo “hi”; sleep 20; done
(2) 如何实现每7分钟运行一次任务?
sleep命令:
sleep NUMBER[SUFFIX]…
SUFFIX:
s: 秒, 默认 (支持0.1秒 、0.01秒)
usleep 1000 1毫秒
usleep 1000000 1秒
m: 分
h: 小时
d: 天

练习
1、每周的工作日1:30,将/etc备份至/backup目录中,保存的文件名称格式为“etcbak-yyyy-mm-dd-HH.tar.xz”,其中日期是前一天的时间

2、每两小时取出当前系统/proc/meminfo文件中以S或M开头的信息追加至/tmp/meminfo.txt文件中

3、工作日时间,每10分钟执行一次磁盘空间检查,一旦发现任何分区利用率高于80%,就执行wall警报



流程控制
过程式编程语言:
顺序执行
选择执行
循环执行

1.条件选择if语句
选择执行:
注意:if语句可嵌套
单分支
if 判断条件;then
条件为真的分支代码
fi
单分支相当于cmd1 && cmd2

双分支
if 判断条件; then
条件为真的分支代码
else
条件为假的分支代码
fi
逐条件进行判断,第一次遇为“真”条件时,执行其分支,而后结束整个if语句
双分支相当于 cmd1 && cmd2 || cmd3 但是cmd2必须成功

多分支
if 判断条件1; then
条件1为真的分支代码
elif 判断条件2; then
条件2为真的分支代码
elif 判断条件3; then
条件3为真的分支代码
else
以上条件都为假的分支代码
fi

逐条件进行判断,第一次遇为“真”条件时,执行其分支,而后结束整个if语句

If示例
1.根据命令的退出状态来执行命令
if ping -c1 -W2 station1 &> /dev/null; then
echo ‘Station1 is UP’
elif grep “station1” ~/maintenance.txt &> /dev/null; then
echo ‘Station1 is undergoing maintenance‘
else
echo ‘Station1 is unexpectedly DOWN!’
exit 1
fi

2.age.shell输入一个年龄数值,显示相应的信息(18岁以下 好好学习天天向上,60岁以下努力工作,60岁以后享受生活,其他输入你不是地球人)

3.写一个脚本areyourich.sh 完成如下逻辑

4.条件判断:case语句

case示例:
1.写个脚本,numcase.sh 输入一个数字,属于1、2、3就显示1,2,3;属于4,5,6就显示4,5,6;属于7,8,9就显示7,8,9;否则就显示“other digit”
read -p “Please input a digit: ” num
case $num in
1|2|3)
echo 1,2,3
;;
4|5|6)
echo 4,5,6
;;
7|8|9)
echo 7,8,9
;;
*)
echo “other digit”
;;
esac

2.写个脚本,menucase.sh。列出菜单点菜
001.rice–2 RMB
002.noodles–15 RMB
003.dumpligs—18 RMB
004.beer— 6RMB

练习
1、编写脚本/root/bin/createuser.sh,实现如下功能:使用一个用户名做为参数,如果指定参数的用户存在,就显示其存在,否则添加之;显示添加的用户的id号等信息

2、编写脚本/root/bin/yesorno.sh,提示用户输入yes或no,并判断用户输入的是yes还是no,或是其它信息

3、编写脚本/root/bin/filetype.sh,判断用户输入文件路径,显示其文件类型(普通,目录,链接,其它文件类型)

4、编写脚本/root/bin/checkint.sh,判断用户输入的参数是否为正整数
正整数:[[ “ n u m "=   [ 1 9 ] [ 0 9 ] + ]]

5.循环–循环条件已知的
循环执行
将某代码段重复运行多次
重复运行多少次
循环次数事先已知
循环次数事先未知
有进入条件和退出条件
for, while, until

6.for循环语法一:
for 变量名 in 列表;do
循环体
done
执行机制:
依次将列表中的元素赋值给“变量名”; 每次赋值后即执行一次循环体; 直到列表中的元素耗尽,循环结束

for循环示例
1.写个脚本sum100-for.sh,,先列出1-100的数字,然后求1-100累加和
for num in {1..100}; do echo “num= n u m ; d o n e s u m = 0 ; f o r n u m i n 1..100 ; d o s u m = [sum+num]; done;echo sum=$sum

列表生成方式:
(1) 直接给出列表
(2) 整数列表:
(a) {start..end}
(b) (seq [start [step]] end)  seq 1 2 10   #1-10,步进值=2  (3) 返回列表的命令 (COMMAND)
(4) 使用glob(通配符),如:*.sh
(5) 变量引用;
@ *

列表生成脚本示例:
1.写脚本creatuser100-magedu.sh,创建user1…..user100 ,初始密码为magedu,并且首次登录必须修改密码
for i in seg 1 100;do
useradd user i &>/dev/null  echo magedu | passwd –stdin user i &>/dev/null
passwd -e user i >/dev/null
done

2.写脚本scanip.sh,扫描当前网段哪些IP是up的,并记录到文件./ipist.log,顺序执行太慢,所以要用并发执行{}&

./iplist.log #每次追加IP,需要先清空一次免得重复
read -p “Please input network ID (eg:192.168.0.0): ” net
net=echo $net | cut -d. -f1-3 #用户自定义
for i in seq 1 254;do
{if ping -c1 -W1 n e t . i &> /dev/null; then
echo “The host n e t . i is up”
echo n e t . i > ./iplist.log
else echo “The host n e t . i is down”
fi} &
done
wait #wait命令,当脚本执行完之后自动回车,显示提示符免得执行完了却不出现提示符

3.写个脚本netid.sh,让用户输入IP地址和子网掩码,计算出网络ID。
read -p “Please input a ip address: ” ip
read -p “Please input a netmask: ” netmask
for i in {1..4} ; do
net=echo $ip | cut -d. -f$i
mask=echo $netmask | cut -d. -f$i
if [ i e q 1 ] ; t h e n n e t i d = [net&mask]
else
netid= n e t i d . [net&mask]
fi
done
echo Network ID is $netid

练习:用for实现
1、判断/var/目录下所有文件的类型

2、添加10个用户user1-user10,密码为8位随机字符

3、/etc/rc.d/rc3.d目录下分别有多个以K开头和以S开头的文件;分别读取每个文件,以K开头的输出为文件加stop,以S开头的输出为文件名加start,如K34filename stop S66filename start

4、编写脚本,提示输入正整数n的值,计算1+2+…+n的总和

5、计算100以内所有能被3整除的整数之和

6、编写脚本,提示请输入网络地址,如192.168.0.0,判断输入的网段中主机在线状态

7、打印九九乘法表

8、在/testdir目录下创建10个html文件,文件名格式为数字N(从1到10)加随机8个字母,如:1AbCdeFgH.html

9、打印等腰三角形

7.while循环
while CONDITION; do
循环体
done

CONDITION:循环控制条件;进入循环之前,先做一次判断;每一次循环之后会再次做判断;条件为“true”,则执行一次循环;直到条件测试状态为“false”终止循环

因此:CONDTION一般应该有循环控制变量;而此变量的值会在循环体不断地被修正
进入条件:CONDITION为true
退出条件:CONDITION为false

猜你喜欢

转载自blog.csdn.net/weixin_40647174/article/details/82343481