launchctl 是一個(gè)統(tǒng)一的服務(wù)管理框架筝野,可以啟動(dòng)、停止和管理進(jìn)程沧奴、應(yīng)用程序痘括、腳本等。
launchctl 可以通過(guò) plist 文件來(lái)指定執(zhí)行周期性任務(wù)滔吠。
查看已存在的任務(wù)
- ~/Library/LaunchAgents :由用戶自己定義的任務(wù)項(xiàng)
- /Library/LaunchAgents :由管理員為用戶定義的任務(wù)項(xiàng)
- /Library/LaunchDaemons :由管理員定義的守護(hù)進(jìn)程任務(wù)項(xiàng)
- /System/Library/LaunchAgents :由Mac OS X為用戶定義的任務(wù)項(xiàng)
- /System/Library/LaunchDaemons :由Mac OS X定義的守護(hù)進(jìn)程任務(wù)項(xiàng)
看下電腦中一些應(yīng)用任務(wù):
上面這些就是目前系統(tǒng)中纲菌,一些程序的服務(wù)。
我們隨便點(diǎn)看查看一些其中的任務(wù)配置:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple Computer//DTD PLIST 1.0//EN"
"http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>Label</key>
<string>org.getlantern</string>
<key>ProgramArguments</key>
<array>
<string>/Applications/Lantern.app/Contents/MacOS/lantern</string>
<string>-startup</string>
</array>
<key>RunAtLoad</key>
<true/>
</dict>
</plist>
解讀:
Label
對(duì)應(yīng)的 org.getlantern
表示任務(wù)名稱疮绷,必須唯一翰舌。
ProgramArguments
表示程序參數(shù),數(shù)組的形式冬骚,第一個(gè)為 需要執(zhí)行的程序或者腳本等椅贱,這里的 /Applications/Lantern.app/Contents/MacOS/lantern
和 -startup
意味著打開(kāi)程序 lantern
。
RunAtLoad
是個(gè)布爾值只冻,表示開(kāi)啟自啟項(xiàng)庇麦。
因此,該任務(wù)配置表示:設(shè)置 lantern 應(yīng)用為開(kāi)機(jī)自起項(xiàng)喜德。
實(shí)例
下面我們舉個(gè)簡(jiǎn)單的例子演示一下山橄,這里執(zhí)行一份 python 腳本,保存當(dāng)前執(zhí)行的時(shí)間舍悯。
首先我們需要準(zhǔn)備需要執(zhí)行的腳本任務(wù)航棱。
#!/usr/bin/env python
# -*- coding:utf-8 -*-
import datetime
def add(path, content):
with open(path,'a') as f:
f.write(content + '\n')
path = '/Users/luoliang/Documents/項(xiàng)目/PythonProject/Test/content.txt'
now_time = "{}".format(datetime.datetime.now())
add(path, now_time)
然后我們編寫(xiě) 任務(wù)配置 plist 文件。
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<!-- 名稱贱呐,要全局唯一 -->
<key>Label</key>
<string>com.test.task</string>
<!-- 腳本任務(wù) -->
<key>Program</key>
<string>/Users/luoliang/Documents/項(xiàng)目/PythonProject/Test/task.py</string>
<!-- 運(yùn)行間隔丧诺,與StartCalenderInterval使用其一,單位為秒 -->
<key>StartInterval</key>
<integer>30</integer>
<!-- 錯(cuò)誤輸出 -->
<key>StandardErrorPath</key>
<string>/Users/luoliang/Desktop/errorlog</string>
</dict>
</plist>
最后奄薇,我們需要將任務(wù)配置放到任務(wù)目錄中驳阎,(這里建議在:~/Library/LaunchAgents )然后啟動(dòng)任務(wù)。
開(kāi)啟終端馁蒂,進(jìn)入到對(duì)應(yīng)任務(wù)配置目錄:
// 加載任務(wù)
$ launchctl load com.test.task.plist // 進(jìn)入該目錄后執(zhí)行
查看任務(wù)是否添加:
// 查看任務(wù)列表
$ launchctl list
這樣呵晚,我們的腳本就可以按照指定時(shí)刻被系統(tǒng)執(zhí)行了。
我們檢查一下腳本輸出目錄文件:
上圖可以看出沫屡,腳本正在以每30秒的間隔不斷的被執(zhí)行饵隙。兩次執(zhí)行的時(shí)間在1秒內(nèi)。
launchctl 的一些命令:
// 加載任務(wù)
$ launchctl load com.test.task.plist
// 刪除任務(wù)
$ launchctl unload com.test.task.plist
// 查看任務(wù)列表, 使用 grep '任務(wù)部分名字' 過(guò)濾
$ launchctl list | grep 'com.test.task.plist'
// 開(kāi)始
$ launchctl start com.test.task.plist
// 停止
$ launchctl stop com.test.task.plist
plist 文件的字段說(shuō)明
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<!-- 名稱沮脖,要全局唯一 -->
<key>Label</key>
<string>com.uniflor.notifier</string>
<!-- 要運(yùn)行的程序金矛, 如果省略這個(gè)選項(xiàng)芯急,會(huì)把ProgramArguments的第一個(gè)
元素作為要運(yùn)行的程序 -->
<key>Program</key>
<string>/Users/uniflor/script.sh</string>
<!-- 命令, 第一個(gè)為命令驶俊,其它為參數(shù)-->
<key>ProgramArguments</key>
<array>
<string>/Users/uniflor/script.sh</string>
</array>
<!-- 運(yùn)行時(shí)間 -->
<key>StartCalendarInterval</key>
<dict>
<key>Minute</key>
<integer>30</integer>
<key>Hour</key>
<integer>9</integer>
<key>Day</key>
<integer>1</integer>
<key>Month</key>
<integer>5</integer>
<!-- 0和7都指星期天 -->
<key>Weekday</key>
<integer>0</integer>
</dict>
<!-- 運(yùn)行間隔娶耍,與StartCalenderInterval使用其一,單位為秒 -->
<key>StartInterval</key>
<integer>30</integer>
<!-- 標(biāo)準(zhǔn)輸入文件 -->
<key>StandardInPath</key>
<string>/Users/uniflor/run-in.log</string>
<!-- 標(biāo)準(zhǔn)輸出文件 -->
<key>StandardOutPath</key>
<string>/Users/uniflor/Bin/run-out.log</string>
<!-- 標(biāo)準(zhǔn)錯(cuò)誤輸出文件 -->
<key>StandardErrorPath</key>
<string>/Users/uniflor/Bin/run-err.log</string>
</dict>
</plist>
1饼酿、Label:對(duì)應(yīng)的需要保證全局唯一性榕酒;
2、Program:要運(yùn)行的程序故俐;
3想鹰、ProgramArguments:命令語(yǔ)句
4、StartCalendarInterval:運(yùn)行的時(shí)間药版,單個(gè)時(shí)間點(diǎn)使用dict辑舷,多個(gè)時(shí)間點(diǎn)使用 array <dict>
5、StartInterval:時(shí)間間隔刚陡,與StartCalendarInterval使用其一惩妇,單位為秒
6、StandardInPath筐乳、StandardOutPath、StandardErrorPath:標(biāo)準(zhǔn)的輸入輸出錯(cuò)誤文件乔妈,這里建議不要使用 .log 作為后綴蝙云,會(huì)打不開(kāi)里面的信息。
補(bǔ)充說(shuō)明
1路召、權(quán)限問(wèn)題勃刨。
有時(shí)候,腳本需要改成可執(zhí)行的權(quán)限股淡,例如:
$ chmod 777 task.py
2身隐、腳本需正常編譯
在上述的例子中,python腳本文件最開(kāi)始的部分有兩行說(shuō)明:
#!/usr/bin/env python // 聲明編譯環(huán)境唯灵,即指定編譯器
# -*- coding:utf-8 -*- // 編碼問(wèn)題
如果你的腳本本身就無(wú)法編譯通過(guò)贾铝,更不要說(shuō)系統(tǒng)去執(zhí)行了。
如果你的也是python腳本埠帕,并且腳本文件本身在項(xiàng)目中垢揩,有時(shí)候爆出找不到自身目錄的錯(cuò)誤,那么請(qǐng)保證上面的聲明之后敛瓷,將你的目錄移至到 site-packages 中去叁巨。
另外,如果你的電腦中存在多個(gè)版本的編譯器呐籽,請(qǐng)將 #!/usr/bin/env python
替換為腳本對(duì)應(yīng)的編譯器的絕對(duì)路徑锋勺,否則會(huì)出現(xiàn)因編譯器版本產(chǎn)生的語(yǔ)法錯(cuò)誤蚀瘸。
3、驗(yàn)證腳本的正確性
你可以將執(zhí)行時(shí)間設(shè)置為較近的時(shí)間庶橱,也可以使用下面語(yǔ)句直接執(zhí)行一次腳本:
// 開(kāi)始
$ launchctl start com.test.task.plist
參考
1苍姜、Mac中的定時(shí)任務(wù)利器:launchctl