通過 cron 子系統(tǒng)在任何時候運(yùn)行任務(wù),輕松實現(xiàn)常規(guī)維護(hù)
要想實現(xiàn)連續(xù)不停的計算,任務(wù)必須能夠在任何時候運(yùn)行,
使用cron進(jìn)行自動維護(hù)
。您可以在晚上定時醒來,在幾十臺機(jī)器上登錄并輸入各種命令;也可以放心地休息,而把工作交給 cron。cron 是一個守護(hù)進(jìn)程,也就是不間斷運(yùn)行的進(jìn)程,它根據(jù)調(diào)度計劃執(zhí)行命令。無論白天還是黑夜,無論工作多么繁重,cron 都會毫無怨言地注意時間并按時運(yùn)行作業(yè)。學(xué)習(xí)如何配置和維護(hù) cron,了解它的眾多用途。開始之前
本節(jié)解釋本教程講授什么內(nèi)容,以及如何從中獲得最大的收益。
關(guān)于本教程
本教程解釋如何配置和維護(hù) cron,cron 是幾乎所有 UNIX® 計算機(jī)上都有的作業(yè)調(diào)度程序。另外,本教程演示 cron 的其他一些應(yīng)用。
目標(biāo)
學(xué)習(xí)如何創(chuàng)建、調(diào)度和管理 cron 作業(yè),如何通過定義時間表控制作業(yè)頻率,從每分鐘一次到每年一次。另外,學(xué)習(xí)如何限制訪問 cron 以避免濫用,以及如何結(jié)合使用其他實用程序和 cron 以自動執(zhí)行常見的維護(hù)任務(wù)。
前提條件
本教程針對 UNIX(和類 UNIX)系統(tǒng)的用戶和系統(tǒng)管理員。要想學(xué)習(xí)本教程,您應(yīng)該熟悉命令行 shell 和 shell 腳本。具備在 UNIX 上安裝和配置軟件的經(jīng)驗也是有幫助的。
系統(tǒng)需求
要想運(yùn)行本教程中的示例,需要一臺 UNIX 計算機(jī)。如果希望配置系統(tǒng)范圍的 cron,還需要根訪問權(quán)。本教程中的示例基于在現(xiàn)代 UNIX 系統(tǒng)上廣泛使用的 Vixie cron,并使用 Ubuntu Desktop Linux® version 8.04.1。其他版本的 cron 與 Vixie 相似;具體信息請參見自己的 UNIX 系統(tǒng)文檔。
不中斷運(yùn)行
與您和我不同,計算機(jī)可以永不疲倦地工作,以相同的精力執(zhí)行每個任務(wù)。另外,計算機(jī)可以每周 7 天每天 24 小時連續(xù)工作,包括風(fēng)雪天和公共假日。
cron 守護(hù)進(jìn)程
但是,要想實現(xiàn)連續(xù)不停的計算,任務(wù)必須能夠在任何時候運(yùn)行。您可以晚上定時醒來,在幾十臺機(jī)器上登錄并輸入各種命令;也可以放心地休息,把工作交給 cron。cron 是一個守護(hù)進(jìn)程(也就是不間斷運(yùn)行的進(jìn)程),它根據(jù)調(diào)度計劃執(zhí)行命令。無論白天還是黑夜,無論工作多么繁重,cron 都會毫無怨言地注意時間并按時運(yùn)行作業(yè)。
需要監(jiān)視 FTP 收存箱(drop-box)接收到的數(shù)據(jù)嗎?使用 cron 每幾分鐘運(yùn)行一個 shell 腳本。需要刪除每天積累的草稿文件嗎?在午夜執(zhí)行垃圾收集。希望定期循環(huán)使用日志文件嗎?設(shè)置一個每周調(diào)度計劃。
在這里,學(xué)習(xí)如何配置和維護(hù) cron 并了解它的眾多應(yīng)用。具體地說,本教程討論 Vixie cron,Vixie 這個名稱源自它的開發(fā)者 Paul Vixie。在 FreeBSD、Apple Mac OS X、大多數(shù)風(fēng)格的 Linux 和其他 UNIX 系統(tǒng)上都可以找到 Vixie cron。要想查明您的系統(tǒng)是否運(yùn)行 Vixie cron,可以輸入 man cron 或 man crontab,查看末尾是否提到開發(fā)者是 Paul Vixie。
要想學(xué)習(xí)本教程和使用 cron,您應(yīng)該熟悉至少一種文本編輯器,比如 vi 或 Emacs,還應(yīng)該有使用 UNIX 命令行、shell 腳本和 shell 環(huán)境變量的經(jīng)驗。另外,如果希望修改系統(tǒng)范圍的 cron 配置文件,還必須有根(超級用戶)訪問權(quán)。
了解 cron 概念
cron 守護(hù)進(jìn)程是一個由實用程序和配置文件組成的小型子系統(tǒng),在幾乎所有類 UNIX 系統(tǒng)上都可以找到某種風(fēng)格的 cron。cron 的組件包括守護(hù)進(jìn)程本身;一組系統(tǒng)范圍的配置文件;一組針對特定用戶的配置文件;一個用來添加、修改和刪除用戶配置文件的實用程序;以及一個簡單的訪問控制設(shè)施。一般來說,cron 配置文件或 cron 作業(yè)的列表被稱為 crontab 或 cron 時間表。
守護(hù)進(jìn)程 cron 連續(xù)運(yùn)行,每分鐘檢查一次配置文件中的修改。cron 讀取系統(tǒng)范圍的和針對用戶的 crontab(分別在下面兩段中詳細(xì)說明)、相應(yīng)地更新事件調(diào)度計劃并執(zhí)行這一分鐘內(nèi)應(yīng)該執(zhí)行的所有命令。這個守護(hù)進(jìn)程還捕捉每個作業(yè)的輸出(如果有輸出的話),并把結(jié)果通過電子郵件發(fā)送給作業(yè)的所有者。
可以在三個位置定義與系統(tǒng)相關(guān)的 作業(yè):/etc/crontab、/etc/cron.d 中的任何文件以及特殊目錄 /etc/cron.hourly、/etc/cron.daily、/etc/cron.weekly 和 /etc/cron.monthly:
主要的系統(tǒng) crontab 是/etc/crontab。這個文件有獨(dú)特的語法(在下一節(jié)中討論),其中定義的每個作業(yè)根據(jù)它自己的時間表(比如每小時兩次或每天一次)作為指定的用戶運(yùn)行。使用 /etc/crontab 調(diào)度各種管理和維護(hù)任務(wù)。
還可以在 /etc/cron.d 目錄中維護(hù)一組 crontab。通過創(chuàng)建 crontab,按照邏輯對屬于某一子系統(tǒng)的命令進(jìn)行分組。例如,PHP 5 編程語言的包在 /etc/cron.d 中安裝一個名為 php5 的 crontab,它會定期清除不使用的會話。/etc/cron.d 中的文件采用與 /etc/crontab 相同的語法,每個作業(yè)按照自己的時間表并作為特定的用戶運(yùn)行。
還可以把 shell 腳本直接放在 /etc/cron.hourly、/etc/cron.daily、/etc/cron.weekly 或 /etc/cron.monthly 目錄中,這樣就可以每小時、每天、每周或每月運(yùn)行此腳本一次。放在這里的腳本作為超級用戶運(yùn)行。
針對用戶的 crontab 集合通常放在 /var/spool/cron/crontabs 中。(具體位置請參見您的 UNIX 系統(tǒng)文檔。一些系統(tǒng)把用戶 crontab 放在 /usr/lib 中)。但是,不能直接編輯此目錄中的文件,而是使用 crontab 實用程序創(chuàng)建 crontab 并提交文件。稍后討論如何管理個人的 crontab。
最后,可以分別使用訪問控制文件 /etc/cron.allow 和 /etc/cron.deny 允許或拒絕用戶訪問 cron。例如,如果某個用戶的作業(yè)可能會破壞系統(tǒng)的正常操作,就可以拒絕他訪問 cron。
如您所見,不需要為保持系統(tǒng)不間斷運(yùn)行而犧牲您的休息時間。只需確定作業(yè)、定義它的調(diào)度計劃并在適當(dāng)?shù)?crontab 中設(shè)置此作業(yè),然后就可以放心地休息了,F(xiàn)在就來看看 cron 文件的特殊語法。
處理 crontab
crontab 僅僅是一個文本文件,可以用任何 UNIX 編輯器編輯它。它可以包含四種代碼行:空行、注釋、環(huán)境變量設(shè)置和命令。
空行和注釋
文件中的空行和多余的空格被忽略?招泻涂崭裼兄谔岣 crontab 的可讀性,使 crontab 組織有序。
還可以使用注釋對每個作業(yè)的時間表和用途加以說明。要想創(chuàng)建注釋,只需在一行的開頭加上一個井號(#)。
環(huán)境變量和命令
cron 最終使用一個 shell 執(zhí)行每個命令?梢酝ㄟ^環(huán)境變量修改或定制 shell 的行為。
在 crontab 中很容易設(shè)置 shell 環(huán)境變量。只需輸入 VARIABLE=value,把 VARIABLE 替換為變量名,把 value 替換為一個值。例如,crontab 行:
PATH=/usr/bin:/bin:/usr/local/bin
指定一個有序的目錄列表作為 shell 搜索路徑。
cron 預(yù)定義了五個環(huán)境變量:
PATH 的默認(rèn)值是 /usr/bin:/bin。
SHELL 預(yù)設(shè)置為 /bin/sh。
LOGNAME 初始化為 crontab 所有者的用戶名。
HOME 設(shè)置為 crontab 所有者的主目錄,比如 /home/joe。
MAILTO 設(shè)置為 crontab 所有者的名稱。
要想修改這些默認(rèn)值或設(shè)置任何變量,只需在 crontab 中設(shè)置適當(dāng)?shù)沫h(huán)境變量。
當(dāng)然,crontab 可以包含任意數(shù)量的命令行。每個命令行指定一個頻率、一個用戶名(只對于系統(tǒng) crontab)和一個要運(yùn)行的任務(wù)。例如,命令:
5 0 * * * root find /tmp -type f -empty -delete
在每天 12:05 a.m(5 0 * * *)刪除 /tmp 中的所有空文件和目錄(find /tmp -type f -empty -delete)。此作業(yè)作為根用戶(root)運(yùn)行。
系統(tǒng) crontab 命令必須指定一個用戶名,作為運(yùn)行此任務(wù)所用的用戶。(因此,在 /etc/crontab 中可能看到上面的命令)。針對用戶的 crontab 不能指定用戶名;一個用戶的 cron 命令總是作為這個用戶運(yùn)行。是否有用戶名是系統(tǒng) crontab 和用戶 crontab 之間的惟一差異。
下面討論定制時間表的眾多方式。
定制時間表
cron 時間表允許以多種方式運(yùn)行作業(yè),比如每分鐘或在特定日期的特定時間。調(diào)度的參數(shù)非常靈活。
cron 字段
可以通過五個字段調(diào)整頻率:分、小時、月中日、月份和周中的日期(如周一,周二)。表 1 總結(jié)如何調(diào)整每個字段。
表 1. cron 作業(yè)的調(diào)度選項
位置 字段 值 說明 1 分 0-59 2 小時 0-23 3 月中日 1-31 與分和小時不同,月中日不是從零開始的。 4 月份 1-12 月份也不是從零開始的。也可以不使用 1-12 的數(shù)字,而是使用月份名的前三個字母,比如 jan 或 may。 5 周中日 0-7 0 和 7 都代表星期日。還可以使用名稱的前三個字母,比如 mon 或 wed。
除了名稱或數(shù)字之外,還可以使用星號(*)表示 “所有”。例如,在分鐘位置上的星號表示這一天中的每一分鐘。(在某些情況下確實需要如此高的頻率,但是一定要小心,以這種頻率執(zhí)行的任務(wù)應(yīng)該非常簡單,不會長時間運(yùn)行)。
還可以使用值的列表、范圍和步長(增量)分別指定多個值、連續(xù)的值范圍和不連續(xù)的值范圍。甚至可以組合使用列表和范圍。列表 是一個逗號分隔的值集。范圍 由啟始值和結(jié)束值(含)以及可選的步長值構(gòu)成。
我們來看一些示例。表 2 中的每一行包含一個時間表及其說明。當(dāng)分、小時和日期字段與當(dāng)前時間匹配時,cron 會執(zhí)行命令;如果月中日和周中日受到限制(也就是說,不是 *),那么當(dāng)這兩個字段中至少一個 與當(dāng)前時間匹配時,cron 也會執(zhí)行命令。
表 2. cron 作業(yè)的時間表示例
調(diào)度計劃 分 小時 月中日 月份 周中日期 說明 0 1 15 1,3,5,7,9,11 * 在 1、3、5、7、9 和 11 月的 15 日的 1 a.m. 運(yùn)行命令。為了更容易理解,也可以把這個時間表寫成 0 1 15 jan,mar,may,jul,sep,nov *。在指定列表時,不要在逗號后面加空格。 0-59/15 * * * * 這個調(diào)度計劃每 15 分鐘運(yùn)行命令一次。 30 * * * wed,fri 這個時間表只在星期三和星期五每小時的 30 分時執(zhí)行命令一次。(在列表中可以使用日和月份的名稱,但是在范圍中不可以)。 0,30 0-5,17-23 * * * 在午夜到 5 a.m. 以及 7 p.m. 到 11 p.m. 之間整點(diǎn)時和 30 分時運(yùn)行命令。 0 0 1 1 * 在每年 1 月 1 日午夜執(zhí)行命令一次。 0 0 * * 0 在每個星期日午夜運(yùn)行命令。這相當(dāng)于每周一次。 30 0 10,20,30 * 6 因為月中日和周中日受到限制,這個時間表在每星期六和每月的 10、20 和 30 日(二月除外)的 12:30 a.m. 運(yùn)行命令。如您所見,實際上可以使用這五個參數(shù)指定任何調(diào)度計劃。為了更加簡便,Vixie cron 還提供了常用調(diào)度計劃的簡寫形式。表 3 列出一些簡寫形式。
表 3. 常用調(diào)度計劃的簡寫形式
簡寫形式 說明 @reboot 每當(dāng)計算機(jī)重新引導(dǎo)時運(yùn)行命令。 @daily 每天一次的簡寫形式。 @weekly 每周一次的簡寫形式。 @annually 每年一次的簡寫形式。也可以寫成 @yearly。 @midnight 在每天午夜運(yùn)行命令。這個簡寫形式相當(dāng)于 @daily。如果喜歡使用簡寫形式,只需用它們替代 cron 命令的前五個字段。下面這個命令看起來簡單多了。
@daily root /usr/local/scripts/clean_old_files.sh
crontab 命令示例
掌握了基本概念之后,我們來看一些用戶 crontab 命令示例。同樣的命令也可以應(yīng)用系統(tǒng)范圍:只需在所有系統(tǒng) crontab 項中周中日字段(第五個字段)后面指定一個用戶名。
創(chuàng)建個人 crontab
要想創(chuàng)建個人 crontab,可以用任何文本編輯器創(chuàng)建一個文件。按照慣例,個人 crontab 文件保存在 ~/.crontab 中,但是可以使用任何文件名。
PATH=/usr/bin:/bin:/usr/local/bin # # Every day, print and delete all temporary files whose names begin with '.#' @daily find $HOME -type f -name '.#*' -print -delete # # Every week, show me what is consuming space in my home directory @weekly du -sh $HOME
通過 crontab 實用程序提交個人 crontab
在編輯文件(比如 ~/mycrontab)之后,通過 crontab 實用程序把它提交給 cron:
% crontab ~/mycrontab
查看 cron 中存儲的信息
要想查看 cron 中存儲的信息,可以輸入 crontab -l:
% crontab -l PATH=/usr/bin:/bin:/usr/local/bin # # Every day, print and delete all temporary files whose names begin with '.#' @daily find $HOME -type f -name '.#*' -print -delete # # Every week, show me what is consuming space in my home directory @weekly du -sh $HOME
替換 crontab
在任何時候,都可以使用 crontab 實用程序替換您的 crontab,
電腦資料
《使用cron進(jìn)行自動維護(hù)》(http://m.clearvueentertainment.com)。只需提交一個新文件或同一文件的修訂版。要想刪除 crontab 作業(yè),只需輸入 crontab -r:% whoami joe % crontab ~/mycrontab % crontab -l PATH=/usr/bin:/bin:/usr/local/bin ... % crontab -r % crontab -l crontab: no crontab for joe
替代 cron 的機(jī)制
盡管 cron 確實很有用,但是您還應(yīng)該了解可以替代它的兩種機(jī)制。
anacron
如果系統(tǒng)常常關(guān)機(jī)或進(jìn)入休眠狀態(tài)(例如,如果使用 UNIX 筆記本計算機(jī)的話),那么可以考慮在系統(tǒng)中添加 anacron。anacron 與 cron 的相似之處在于,它也把作業(yè)安排在以后運(yùn)行;但是,與 cron 不同,即使作業(yè)的預(yù)定運(yùn)行時間已經(jīng)過了,anacron 也會運(yùn)行作業(yè)。
例如,如果安排在星期六運(yùn)行文件系統(tǒng)備份,但是系統(tǒng)在星期五到星期一關(guān)機(jī)了,那么當(dāng)系統(tǒng)在星期一重新啟動時,anacron 會立即運(yùn)行星期六的作業(yè)。與之相反,cron 僅僅檢查現(xiàn)在是否應(yīng)該運(yùn)行作業(yè);因此,如果在作業(yè)的預(yù)定運(yùn)行時間系統(tǒng)是關(guān)閉的,就不運(yùn)行作業(yè)。
anacron 的調(diào)度選項比 cron 少得多。它只能以整天的時間間隔調(diào)度作業(yè),比如一天、7 天或 30 天;但是對于那些必須頻繁且可靠地運(yùn)行的作業(yè),它是更好的選擇。
另外,必須從 cron 啟動 anacron。每當(dāng) anacron 運(yùn)行時,它讀取自己的配置文件。配置文件包含由作業(yè)及其頻率(用天數(shù)表示)組成的配置對。如果作業(yè)在預(yù)定的時間沒有運(yùn)行,anacron 就運(yùn)行此作業(yè)并記錄運(yùn)行作業(yè)的時間。運(yùn)行完所有作業(yè)之后,anacron 退出。
在大多數(shù) Linux 發(fā)行版上都可以找到 anacron,但是也很容易自己下載并構(gòu)建源代碼。訪問 anacron 項目頁面 獲得最新版本。
anacron 的主要配置文件可以在 /etc/anacron 中找到?梢韵衽渲 cron 時那樣設(shè)置環(huán)境變量,但是更簡單:
SHELL=/bin/zsh PATH=/usr/bin:/bin:/usr/local/bin # format: frequency delay name job 1 10 day-to-day daily.chores.sh
第一個數(shù)字是周期,所以 1 表示每天運(yùn)行一次,7 表示每 7 天運(yùn)行一次,等等。第二個數(shù)字是延遲,也就是從 anacron 啟動之后到執(zhí)行這個作業(yè)之間等待的分鐘數(shù)。如果把延遲字段設(shè)置為不同的值,就可以防止所有作業(yè)同時啟動。名稱 day-to-day 只是一個有幫助的昵稱。配置行的其余部分指定作業(yè);在這里,每天運(yùn)行在指定路徑中找到的 shell 腳本 daily.chores.sh 一次。
anacron 以手冊頁形式提供了出色的文檔,還可以在網(wǎng)上找到關(guān)于 anacron 的提示。(請閱讀我在 2007 年 10 月編輯的 Rod Smith 撰寫的 Linux Magazine 文章)。anacron 適合 UNIX 愛好者和需要額外保障的系統(tǒng)管理員使用。
launchd:cron 的現(xiàn)代替代機(jī)制
cron 確實是一種功能強(qiáng)大、值得信賴的實用程序,它的廣泛流行就證明了這一點(diǎn)。Vixie cron 最近又有所改進(jìn),比如增加了簡寫方式 @reboot,進(jìn)一步簡化了管理。但是,cron 仍然有一些缺點(diǎn):
盡管可以在 crontab 文件中定義 cron 作業(yè),但是不能從命令行啟動和停止 cron 作業(yè)。另外,不能在命令行上創(chuàng)建專用作業(yè)并提交給日歷。
cron 不實施資源限制。如果作業(yè)作為根用戶運(yùn)行,就能夠消耗無數(shù)的處理器時間和內(nèi)存。在實踐中,可能希望限制一個作業(yè),以免它影響其他 cron 作業(yè)和系統(tǒng)操作的總體質(zhì)量。
cron 作業(yè)嚴(yán)格地與一個調(diào)度計劃相關(guān)聯(lián)。例如,無法只在發(fā)生某一事件(比如創(chuàng)建一個文件)時啟動作業(yè)。
從更大的范疇來看,類 UNIX 系統(tǒng)的許多核心組件都能夠根據(jù)需要啟動其他程序,包括 cron、用于網(wǎng)絡(luò)守護(hù)進(jìn)程的 xinetd(或 inetd)和 init(所有系統(tǒng)進(jìn)程的起源)。每個核心組件都有自己的配置文件,所以很難知道哪個組件最適合完成某個修改。
為了克服這些缺點(diǎn),Apple Computer 開發(fā)了一個統(tǒng)一的啟動工具 launchd,它可以在引導(dǎo)時、根據(jù)需要和按照指定的時間間隔啟動進(jìn)程。實際上,在 Mac OS 10.4 Tiger 中 launchd 已經(jīng)替代了 cron(和 init 以及用來引導(dǎo)和初始化系統(tǒng)的其他幾個系統(tǒng)實用程序)。(Apple 在系統(tǒng)上保留了 cron,以便為用戶提供方便,而且 Vixie cron 的調(diào)度選項更靈活)。Mac OS X 的引導(dǎo)速度很快確實可以歸功于 launchd:它會在引導(dǎo)時列出要啟動的程序,但是只在首次需要時執(zhí)行程序。
launchd 是開放源碼的,可以從 Mac OS Forge 上它的主頁獲得源代碼。目前,launchd 已經(jīng)被移植到 FreeBSD 上,但是還沒有移植到其他 UNIX 或 Linux 系統(tǒng)。但是,許多項目正在積極地實現(xiàn)與 launchd 等效的程序,所以簡要地了解它的特性是有必要的:
launchd 并不創(chuàng)建作業(yè)來檢查目錄中是否有新文件,而是自動監(jiān)視目錄中是否有新文件或者監(jiān)視空目錄中是否添加了任何文件,并根據(jù)需要啟動作業(yè)。launchd 不執(zhí)行輪詢;而是使用 kqueues 設(shè)施讓內(nèi)核在目錄發(fā)生變化時通知它。(Linux 具有一種相似的事件設(shè)施 inotify,以后的一篇 developerWorks 文章將討論它) 。
如果指定了,launchd 會使用 chroot 把您的作業(yè)發(fā)送到一個新目錄。chroot 讀作 “cha-root”,它是一個系統(tǒng)調(diào)用,可以改變前向斜杠(/)和根目錄指向的目錄。因此,如果使用 chroot 把文件發(fā)送到 /opt/root,/opt/root 之外的所有文件就是不可訪問的(畢竟,/opt/root 現(xiàn)在是 /,也就是文件系統(tǒng)的頂級目錄),/opt/root 中的所有目錄成為一級目錄。通常使用 chroot 限制作業(yè),使代碼無法進(jìn)入更大的文件系統(tǒng),以避免產(chǎn)生損害。
可以為作業(yè)設(shè)置資源限制。可以限制的資源包括內(nèi)存、堆棧大小和打開的文件的最大數(shù)量。
當(dāng)定義一個任務(wù)并把它裝載到 launchd 中之后,可以按照名稱從命令行啟動和停止作業(yè)。
launchd 由三個組件組成:launchd 守護(hù)進(jìn)程本身;用來添加、修改和刪除作業(yè)以及控制 launchd 的 launchctl 實用程序;一個或多個配置文件,每個文件定義一個或多個作業(yè)。由于它起源于 Mac OS X,launchd 配置文件只是簡單的屬性文件,可以用 Extensible Markup Language (XML) 表示。
簡單地說,在 Mac OS X 上使用 launchd 的步驟如下(要監(jiān)視一個目錄中添加的文件并根據(jù)需要運(yùn)行作業(yè)):
1.創(chuàng)建一個屬性文件來表達(dá)此作業(yè)及其所有屬性。
可以使用 Mac 的 Property Editor,也可以手工編輯 XML。無論采用哪種方法,產(chǎn)生的文件都應(yīng)該與 清單 1 相似。
清單 1. 監(jiān)視文件系統(tǒng)目錄變化的 launchd 作業(yè)示例
<?xml version="1.0" encoding="UTF-8"?><br /><!DOCTYPE plist PUBLIC "-//Apple Computer//DTD PLIST <br /> 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"><br /><plist version="1.0"><br /><dict><br /> <key>Label</key><br /> <string>com.example.processor</string><br /> <key>OnDemand</key><br /> <true/><br /> <key>Program</key><br /> <string>/Users/strike/bin/processor</string><br /> <key>ProgramArguments</key><br /> <array><br /> <string>processor</string><br /> </array><br /> <key>WatchPaths</key><br /> <array><br /> <string>/Users/strike/data/incoming</string><br /> </array><br /></dict><br /></plist>
簡單地說,當(dāng) /Users/strike/data/incoming 目錄的內(nèi)容發(fā)生變化時,此文件運(yùn)行 /Users/strike/bin/processor 中的實用程序腳本。把 OnDemand 設(shè)置為 True 讓 launchd 根據(jù)需要啟動此作業(yè)。把此文件保存到 ~/Library/LaunchAgents/com.example.processor.plist。
2.用 launchctl 把此作業(yè)裝載到 launchd 中: % launchctl load ~/Library/LaunchAgents/com.example.processor.plist
如果希望檢查最近的操作或查看保存的作業(yè)的列表,只需輸入 launchctl list。
3.要想刪除作業(yè),只需帶 unload 選項使用 launchctl: % launchctl unload -w ~/Library/LaunchAgents/com.example.processor.plist
-w 的作用是什么?它完全刪除 launchd 中的作業(yè)。如果沒有這個選項,作業(yè)會在登錄時自動地重新裝載(因為作業(yè)還在啟動代理的用戶集合中)。
launchd 手冊頁包含大量信息;如果您是 Mac OS X 用戶,還可以找到大量針對 launchd 的應(yīng)用程序。一些聰明的開發(fā)人員正在把 launchd 遷移到更廣泛的平臺。
使用 cron 讓工作更輕松
學(xué)習(xí)關(guān)于使用 cron 的一些提示和技巧,了解這個守護(hù)進(jìn)程和相似的其他程序為什么是您的好朋友。
提示和技巧
下面是關(guān)于 cron 的一些提示、技巧和常見的問題:
與在終端窗口中使用的 shell 或 shell 腳本不同,cron 并不 在 crontab 文件中就地展開環(huán)境變量。換句話說,如果在 crontab 中輸入: HOME=/home/joe
PATH=$HOME/bin:/usr/bin:/bin
那么 PATH 并不會設(shè)置為您期望的路徑。您必須手工展開所有變量,比如:
HOME=/home/joe
PATH=/home/joe/bin:/usr/bin:/bin
但是,因為每個 cron 命令都由 shell 執(zhí)行,所以命令可以 引用變量名。例如,如果在個人 crontab 中編寫以下命令(注意這一行中省略了用戶名參數(shù)):
@daily $HOME/bin/cleanup_daily.sh
那么 $HOME 會正確地展開。
不要把計算密集型任務(wù)安排在同一時間啟動,比如 @midnight。如果可能的話,在凌晨的幾小時中分散地啟動這些任務(wù),以避免它們爭用資源。
正如上面提到的,在默認(rèn)情況下環(huán)境變量 SHELL 設(shè)置為 /bin/sh。如果不修改此變量,crontab 中的所有命令都由 /bin/sh 解釋。但是,如果您不熟悉 /bin/sh,更喜歡另一種 shell,那么可以設(shè)置 SHELL 并使用這種 shell 的命令語法。
例如,如果設(shè)置 SHELL=/bin/zsh,那么所有命令都可以使用 Z shell 的功能,比如它的高級重定向操作符:
SHELL=/bin/zsh
@daily uptime > daily >> weekly
在這里,uptime 命令的輸出覆蓋 daily 文件(>daily)并追加到 weekly 文件中(>> weekly)。
使用訪問控制列表 (ACL) — /etc/cron.allow 和 /etc/cron.deny — 允許或拒絕特定用戶運(yùn)行 cron 作業(yè)。如果希望把對 cron 的訪問權(quán)限制在很少幾個用戶,那么在 /etc/cron.allow 中列出這些用戶的用戶名。未指定的任何用戶都無法使用 crontab 實用程序提交 crontab。但是,如果希望允許大多數(shù)人訪問 cron 而拒絕少數(shù)用戶,那么在 /etc/cron.deny 中列出受到限制的用戶。
例如,如果 /etc/cron.allow 的內(nèi)容如下:
joe
zelda
那么除 Joe 和 Zelda 之外任何用戶都無法訪問 cron:
% whoami
strike
% crontab ~/.crontab
You (strike) are not allowed to use this program (crontab)
See crontab(1) for more information
要想禁用 cron 發(fā)出的電子郵件報告,應(yīng)該在 crontab 中設(shè)置 MAILTO=""。
再次重申,不要在列表中使用空格。列表值以逗號分隔。在 Vixie cron 中,在范圍中不使用日和月份的名稱。
要仔細(xì)閱讀系統(tǒng)的 cron 文檔。路徑、特性和簡便措施都可能不一樣。在命令行上輸入 man 5 crontab 了解 crontab 文件的語法。輸入 man 1 crontab 了解 crontab 實用程序。在命令行上輸入 man cron 或 man 8 cron 了解 cron 守護(hù)進(jìn)程本身的選項。
系統(tǒng)管理員最好的朋友
cron 和與它相似的程序?qū)τ谙到y(tǒng)管理員非常有幫助。如果您需要反復(fù)執(zhí)行相同任務(wù),就可以考慮用 cron 實現(xiàn)自動化。捕捉具有許多步驟的復(fù)雜任務(wù)常常需要 shell 腳本,但是許多任務(wù)只需要一行命令。
下面僅僅是一些思路:
通過組合使用 cron 和數(shù)據(jù)庫工具,創(chuàng)建每日轉(zhuǎn)儲。例如,命令: @daily joe mysqldump -pjoespwd accounts > $HOME/backups/accounts.`date +%F`.sql
每天把數(shù)據(jù)庫 accounts 轉(zhuǎn)儲到一個文件。嵌入的日期命令(`date +%F`)確保文件名是惟一的,比如 accounts.2008-08-07.sql。此命令作為用戶 joe 運(yùn)行,所以用 -p 指定 Joe 的密碼。此命令還可以放在 Joe 自己的 crontab 中,因為轉(zhuǎn)儲需要他的 MySQL 憑證。
locate 子系統(tǒng)為系統(tǒng)上的所有文件編制索引,并把每個文件的完整路徑存儲在數(shù)據(jù)庫中。然后,從命令行查詢此數(shù)據(jù)庫,就可以立即找到文件。當(dāng)然,可以根據(jù)需要用 find 搜索文件,但是必須等待它搜索文件系統(tǒng),這可能很慢。
為了讓定位子系統(tǒng)發(fā)揮作用,必須定期為文件系統(tǒng)編制索引,因為隨時可能添加和刪除文件。這種情況就非常適合使用 cron。
0 0,12 * * * root updatedb
這個 crontab 項每天運(yùn)行 updatedb(locate 更新實用程序)兩次。
顯然很適合用 cron 實現(xiàn)自動化的另一個任務(wù)是,把文件從主服務(wù)器復(fù)制到眾多的從服務(wù)器。rsync 是一種跨多個系統(tǒng)分布和同步文件集合的現(xiàn)代實用程序。許多網(wǎng)管都通過組合使用 cron 和 rsync 把網(wǎng)站的主拷貝復(fù)制到服務(wù)器群中的每臺服務(wù)器。 @midnight www rsync -avz /var/www/site slave1:/var/www
在每天午夜,rsync 都會把 /var/www/site 復(fù)制(-avz)到 slave1 上的 /var/www。
使用命令行實用程序 mail 和 shell 管道操作符(|)把任務(wù)的輸出發(fā)送給團(tuán)隊中的一個或多個成員。
@weekly root df --print-type --local -h |& mail -s "Weekly df report" andy bob
在這里,每周通過電子郵件把 df 的輸出發(fā)送給用戶 Andy 和 Bob,讓他們能夠監(jiān)視磁盤使用量。
教程結(jié)束語
無論您是單獨(dú)使用 UNIX 系統(tǒng),還是管理有數(shù)百位用戶的系統(tǒng),自動執(zhí)行維護(hù)任務(wù)都有助于節(jié)省時間、減少錯誤以及保持機(jī)器不間斷運(yùn)行。cron 是在 UNIX 系統(tǒng)上實現(xiàn)自動化的關(guān)鍵組件,只需發(fā)揮一點(diǎn)兒想像力,就可以讓計算機(jī)為您工作,而不是由您 “伺候” 計算機(jī)。
cron 有助于更輕松地完成工作,F(xiàn)在,您可以睡個好覺了。