您的位置:首页 > 其它

定时任务 Crontab命令 详解

2015-10-13 10:55 183 查看
前言

crontab是Unix和Linux用于设置周期性被执行的指令,是互联网很常用的技术,很多任务都会设置在crontab循环执行,如果不使用crontab,那么任务就是常驻程序,这对你的程序要求比较高,一个要求你的程序是24X7小时不宕机,一个是要求你的调度程序比较可靠,实际工作中,90%的程序都没有必要花这么多时间和精力去解决上面的两个问题的,只需要写好自己的业务逻辑,通过crond这个工业级程序去调度就行了,crond的可靠性,健壮性,大家应该是毫无疑问的。

crontab简易入门

假设我要设置一个任务,每分钟就要做一个数据同步,这个同步脚本的路径是/home/blue/do/rsyncfile.sh,那么我可以在这么配置,使用blue用户,在终端输入

1
crontab
-e
2
#
此时会进入vi的编辑画面让您编辑工作!注意到,每项工作都是一行。
3
#分
时日月周|<==============任务的完整命令行
4
*
****/home/blue/
do
/rsyncfile.sh
默认情况下,任何使用者只要不被列入/etc/cron.deny当中,那么他就可以直接下达『crontab-e』去编辑自己的例行性命令了!整个过程就如同上面提到的,会进入vi的编辑画面,然后以一个工作一行来编辑,编辑完毕之后输入『:wq』储存后离开
vi就可以了!

假如我们需要修改为每5分钟运行数据同步的脚本,那么同样使用crontab-e进入编辑:

1
*/5
****/home/blue/
do
/rsyncfile.sh
假如服务器出了问题,有一天的数据没有同步,于是我们就需要补数据了,假设这个补数据的脚本是/home/blue/do/rsyncfile_day.sh,但是白天是高峰期,晚上用户不多,是低峰期,我们补数据会占用大量带宽,尤其是白天,会影响正常业务,所以一般我们可以让补数据任务在凌晨2点开始跑,那么同样使用crontab-e进入编辑:

1
0
214*/home/blue/
do
/rsyncfile_day.sh
这样,在4月1号凌晨2点0分就会开始启动我们的补数据的脚本了。

同步数据,在互联网公司是再平常不过的任务了,这里大家可以看到crontab的魅力所在了,只需要写最简单的业务逻辑,把调度交给crond做,就完成了一个可靠性很高的一项任务了,如果要自己去额外写这种调度程序,不知道要花多少精力才能做到可靠稳定。

crontab的语法

1
crontab
[-u
username][-l|-e|-r]
2
选项与参数:
3
-u
:只有root才能进行这个任务,亦即帮其他使用者创建/移除
crontab
工作排程;
4
-e
:编辑
crontab
的工作内容
5
-l
:查阅
crontab
的工作内容
6
-r
:移除所有的
crontab
的工作内容,若仅要移除一项,请用
-e去编辑
查询使用者目前的crontab内容:

1
crontab
-l
2
*/5
****/home/blue/
do
/rsyncfile.sh
3
0
214*/home/blue/
do
/rsyncfile_day.sh
清空使用者目前的crontab:

1
crontab
-r
2
crontab
-l
3
no
crontab
for
blue
如果你想删除当前用户的某一个crontab任务,那么使用crontab-e进入编辑器,再删除对应的任务。

crontab的限制

/etc/cron.allow:将可以使用crontab的帐号写入其中,若不在这个文件内的使用者则不可使用crontab;

/etc/cron.deny:将不可以使用crontab的帐号写入其中,若未记录到这个文件当中的使用者,就可以使用crontab。

以优先顺序来说,/etc/cron.allow比/etc/cron.deny要优先,而判断上面,这两个文件只选择一个来限制而已,因此,建议你只要保留一个即可,免得影响自己在配置上面的判断!一般来说,系统默认是保留/etc/cron.deny,你可以将不想让他运行
crontab的那个使用者写入/etc/cron.deny当中,一个帐号一行!

/etc/crontab配置文件讲解

『crontab-e』是针对使用者的cron来设计的,如果是『系统的例行性任务』时,就要编辑/etc/crontab这个文件。

那就是crontab-e这个crontab其实是/usr/bin/crontab这个运行档,但是/etc/crontab可是一个『纯文字档』,必须用root的身份编辑一下这个文件。

首先我们要来看看crontab的文件内容

01
cat
/etc/
crontab
02
03
#
/etc/crontab:system-widecrontab
04
#
Unlikeanyothercrontabyoudon'thavetorunthe`crontab'
05
#
commandtoinstallthenewversionwhenyoueditthisfile
06
#
andfilesin/etc/cron.d.Thesefilesalsohaveusernamefields,
07
#
thatnoneoftheothercrontabsdo.
08
09
SHELL=/bin/sh
10
PATH=/usr/
local
/sbin:/usr/
local
/bin:/sbin:/bin:/usr/sbin:/usr/bin
11
12
#
mhdommondowusercommand
13
17
****root
cd
/
&&run-parts--report/etc/
cron
.hourly
14
25
6***root
test
-x
/usr/sbin/anacron||(
cd
/
&&run-parts--report/etc/
cron
.daily
)
15
47
6**7root
test
-x
/usr/sbin/anacron||(
cd
/
&&run-parts--report/etc/
cron
.weekly
)
16
52
61**root
test
-x
/usr/sbin/anacron||(
cd
/
&&run-parts--report/etc/
cron
.monthly
)
这个文件与将刚刚我们下达crontab-e的内容几乎完全一模一样!只是有几个地方不太相同

1
PATH=....:
这里就是输入运行档的搜寻路径!使用默认的路径配置就已经很足够了!

1
17
****root
cd
/
&&run-parts--report/etc/
cron
.hourly:
这个/etc/crontab里面预配置义出四项工作任务,分别是每小时、每天、每周及每个月分别进行一次的工作!但是在五个栏位后面接的并不是命令,而是一个新的栏位,那就是『运行后面那串命令的身份』为何!这与使用者的crontab-e不相同。由於使用者自己的crontab并不需要指定身份,但/etc/crontab里面当然要指定身份啦!以上表的内容来说,系统默认的例行性工作是以root的身份来进行的。

那么后面那串命令是什么呢?你可以使用『whichrun-parts』搜寻看看,其实那是一个bashscript啦!如果你直接进入/usr/bin/run-parts去看看,会发现这支命令会将后面接的『目录』内的所有文件捉出来运行!这也就是说『如果你想让系统每小时主动帮你运行某个命令,将该命令写成script,并将该文件放置到/etc/cron.hourly/目录下即可』的意思!

现在你知道系统是如何进行他默认的一堆例行性工作排程了吗?如果你下达『ll/etc/cron.daily』就可以看到一堆文件,那些文件就是系统提供的script,而这堆scripts将会在每天的凌晨6:25开始运行!

假设你现在要作一个目录,让系统可以每2分钟去运行这个目录下的所有可以运行的文件,你可以写下如下的这一行在/etc/crontab中:

1
*/2
****rootrun-parts/etc/
cron
.min
当然罗,/etc/cron.min这个目录是需要存在的喔!那如果我需要运行的是一个『程序』而已,不需要用到一个目录呢?该如何是好?例如在侦测网络流量时,我们希望每五分钟侦测分析一次,可以这样写:

1
*/5
****root/bin/mrtg/etc/mrtg/mrtg.cfg
如何!创建例行性命令很简单吧!如果你是系统管理员而且你的工作又是系统维护方面的例行任务时,直接修改/etc/crontab这个文件即可喔!又便利,又方便管理呢!

crontab的原理

当使用者使用crontab这个命令来创建工作排程之后,该项工作就会被纪录到/var/spool/cron/里面去了,而且是以帐号来作为判别的喔!举例来说,blue使用crontab后,他的工作会被纪录到/var/spool/cron/blue里头去!但请注意,不要使用
vi直接编辑该文件,因为可能由于输入语法错误,会导致无法运行cron喔!另外,cron运行的每一项工作都会被纪录到/var/log/cron这个登录档中,所以罗,如果你的Linux不知道有否被植入木马时,也可以搜寻一下/var/log/cron这个登录档呢!

crond服务的最低侦测限制是『分钟』,所以『cron会每分钟去读取一次/etc/crontab与/var/spool/cron里面的数据内容』,因此,只要你编辑完/etc/crontab
这个文件,并且将他储存之后,那么cron的配置就自动的会来运行了!

备注:在Linux底下的crontab会自动的帮我们每分钟重新读取一次/etc/crontab的例行工作事项,但是某些原因或者是其他的Unix系统中,由于crontab是读到内存当中的,所以在你修改完/etc/crontab之后,可能并不会马上运行,这个时候请重新启动crond这个服务吧!『/etc/init.d/crond
restart』或『servicecrondrestart』

crontab的格式讲解

每项工作(每行)的格式都是具有六个栏位,这六个栏位的意义为:

代表意义分钟小时日期(天)月份命令
数字范围0-590-231-311-120-7呀就命令啊
比较有趣的是那个『周』喔!周的数字为0或7时,都代表『星期天』的意思!另外,
还有一些辅助的字符,大概有底下这些:

特殊字符代表意义
*(星号)代表任何时刻都接受的意思!举例来说,范例一内那个日、月、周都是*,就代表著『不论何月、何日的礼拜几的12:00都运行后续命令』的意思!
,(逗号)代表分隔时段的意思。举例来说,如果要下达的工作是3:00与6:00时,就会是:



03,6***command



时间参数还是有五栏,不过第二栏是3,6,代表3与6都适用!

-(减号)代表一段时间范围内,举例来说,8点到12点之间的每小时的20分都进行一项工作:



208-12***command



仔细看到第二栏变成8-12喔!代表8,9,10,11,12都适用的意思!

/n(斜线)那个n代表数字,亦即是『每隔n单位间隔』的意思,例如每五分钟进行一次,则:



*/5****command



很简单吧!用*与/5来搭配,也可以写成0-59/5,相同意思!

周与日月不可同时并存

另一个需要注意的地方在於:『你可以分别以周或者是日月为单位作为循环,但你不可使用「几月几号且为星期几」的模式工作』。这个意思是说,你不可以这样编写一个工作排程:

1
30
121195root
echo
"just
test"
<==这是错误的写法
本来你以为九月十一号且为星期五才会进行这项工作,无奈的是,系统可能会判定每个星期五作一次,或每年的9月11号分别进行,如此一来与你当初的规划就不一样了~所以罗,得要注意这个地方!上述的写法是不对的!

CentOS下查看crontab执行历史记录

在crontab中添加了定时任务,但发现没有得到期望的结果,因而怀疑是crontab没有执行相应的任务,但怎么定位crontab是否执行呢?

这就需要查看crontab的执行历史记录,具体位置如下:

1
cd
/var/log
2
tail
-100
cron
在cron文件中即可查阅已经操作过的相关定时任务。

参考资料:
http://vbird.dic.ksu.edu.tw/linux_basic/0430cron_3.phphttp://baike.baidu.com/view/1229061.htm
一、Crontab格式说明

我们可以用crontab-e添加要执行的命令。命令执行的结果,无论是标准输出还是错误输出,都将以邮件形式发给用户。

添加的命令必须以如下格式:



*****/commandpath



前五个字段可以取整数值,指定何时开始工作,第六个域是字符串,即命令字段,其中包括了crontab调度执行的命令。各个字段之间用spaces和tabs分割。

前5个字段分别表示:



分钟:0-59

小时:1-23

日期:1-31

月份:1-12

星期:0-6(0表示周日)



还可以用一些特殊符号:



*:表示任何时刻

,:表示分割

-:表示一个段,如第二端里:1-5,就表示1到5点

/n:表示每个n的单位执行一次,如第二段里,*/1,就表示每隔1个小时执行一次命令。也可以写成1-23/1.



一些示例:



008,12,16***/data/app/scripts/monitor/df.sh

302***/data/app/scripts/hotbackup/hot_database_backup.sh

108,12,16***/data/app/scripts/monitor/check_ind_unusable.sh

108,12,16***/data/app/scripts/monitor/check_maxfilesize.sh

108,12,16***/data/app/scripts/monitor/check_objectsize.sh





4321***21:43执行

1505***05:15执行

017***17:00执行

017**1每周一的17:00执行

0,1017**0,2,3每周日,周二,周三的17:00和17:10执行

0-10171**毎月1日从17:00到7:10毎隔1分钟执行

001,15*1毎月1日和15日和一日的0:00执行

4241**毎月1日的4:42分执行

021**1-6周一到周六21:00执行

0,10,20,30,40,50****每隔10分执行

*/10****每隔10分执行

*1***从1:0到1:59每隔1分钟执行

01***1:00执行

0*/1***毎时0分每隔1小时执行

0****毎时0分每隔1小时执行

28-20/3***8:02,11:02,14:02,17:02,20:02执行

3051,15**1日和15日的5:30执行



二、&后台执行命令

当在前台运行某个作业时,终端被该作业占据;而在后台运行作业时,它不会占据终端。可以使用&命令把作业放到后台执行。

如:

1
30
2***/data/app/scripts/hotbackup/hot_database_backup.sh&
在后台运行作业时要当心:需要用户交互的命令不要放在后台执行,因为这样你的机器就会在那里傻等。

不过,作业在后台运行一样会将结果输出到屏幕上,干扰你的工作。如果放在后台运行的作业会产生大量的输出,最好使用下面的方法把它的输出重定向到某个文件中:

如:

1
command
>out.
file
2>&1
&
在这个例子中,2>&1表示所有的标准输出和错误输出都将被重定向到一个叫做out.file的文件中。

三、2>&1含义

先看一个例子:

1
0
2***/u01/
test
.sh
>/dev/null2>&1&
这句话的意思就是在后台执行这条命令,并将错误输出2重定向到标准输出1,然后将标准输出1全部放到/dev/null文件,也就是清空。

在这里有有几个数字的意思:



0表示键盘输入

1表示标准输出

2表示错误输出



我们也可以这样写:



02***/u01/test.sh1>/u01/out.file&

02***/u01/test.sh2>/u01/out.file&

02***/u01/test.sh2>/u01/out.file2>&1&



将tesh.sh命令输出重定向到out.file,即输出内容不打印到屏幕上,而是输出到out.file文件中。

2>&1是将错误输出重定向到标准输出。然后将标准输入重定向到文件out.file。

&1表示的是文件描述1,表示标准输出,如果这里少了&就成了数字1,就表示重定向到文件1。

&:后台执行

测试:



ls2>1:不会报没有2文件的错误,但会输出一个空的文件1;

lsxxx2>1:没有xxx这个文件的错误输出到了1中;

lsxxx2>&1:不会生成1这个文件了,不过错误跑到标准输出了;

lsxxx>out.txt2>&1==lsxxx1>out.txt2>&1:因为重定向符号>默认是1,这句就把错误输出和标准输出都传到out.txt文件中。



四、2>&1写在后面的原因



格式:command>file2>&1==command1>file2>&1



首先是command>file将标准输出重定向到file中,2>&1是标准错误拷贝了标准输出,也就是同样被重定向到file中,最终结果就是标准输出和错误都被重定向到file中。

如果改成:command2>&1>file

2>&1标准错误拷贝了标准输出的行为,但此时标准输出还是在终端。>file后输出才被重定向到file,但标准错误仍然保持在终端。

延伸阅读:

Shell标准输出、标准错误>/dev/null2>&1

如何让Linux定时任务crond以秒为单位执行(如每隔3秒)

经验教训:

打算在服务器上每天晚上23:00定时执行Python脚本,去备份MySql数据库,命令如下:

1
*
23***python/
var
/www/html/crontab_python/back_db.py
>/dev/null2>&1
结果呢,每次备份都产生了60份备份文件,仔细查看定时任务命令,发现在“分”的位置上,少加了个“0”,因为“*”表示该位置的任何一个值,修改如下:

1
0
23***python/var/www/html/crontab_python/back_db.py>/dev/null2>&1


04***/usr/local/php/bin/php/usr/local/nginx/www/backup-db/backup_db.php172.16.8.26>/dev/null2>&1

04***/usr/local/php/bin/php/usr/local/nginx/www/backup-db/backup_db.php172.16.10.151>/dev/null2>&1

转至:http://www.centoscn.com/CentOS/help/2014/1030/4025.html

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: