daemon 音標 : [‘di:mən] , 中文含義為守護神或精靈的意思 . 其實它還有個意思 : 守護進程 .
守護進程簡單地說就是可以脫離終端而在后台運行的進程 . 這在Linux中是非常常見的一種進程 , 比如apache或者mysql等服務啟動后 , 就會以守護進程的方式進駐在內存中 。
以PHP為例 , 假如我有個耗時間的任務需要跑在后台 : 將所有mysql中user表中的2000萬用戶全部導入到redis中做緩存預熱, 那么這個任務估計一時半會是不會結束的 , 這個時候就需要編寫一個php腳本以daemon形式運行在系統中 , 結束后自動退出.
在Linux中 , 大概有三種方式實現腳本后台化 :
1 . 在命令后添加一個&符號 , 比如 php task.php & . 這個方法的缺點在於 如果terminal終端關閉 , 無論是正常關閉還是非正常關閉 , 這個php進程都會隨着終端關閉而關閉 , 其次是代碼中如果有echo或者print_r之類的輸出文本 , 會被輸出到當前的終端窗口中 .
2 . 使用nohup命令 , 比如 nohup php task.php & . 默認情況下 , 代碼中echo或者print_r之類輸出的文本會被輸出到php代碼同級目錄的nohup.out文件中 。
如果你用exit命令或者關閉按鈕等正常手段關閉終端 , 該進程不會被關閉 , 依然會在后台持續運行 。
但是如果終端遇到異常退出或者終止 , 該php進程也會隨即退出 . 本質上 , 也並非穩定可靠的daemon方案 .
3 . 使用fork和setsid , 我暫且稱之為 : *nix解決方案 . 具體看下代碼 : pcntl_fork()
<?php
// 一次fork
$pid = pcntl_fork();
if ($pid < 0) {
exit('fork error.');
} else if ($pid > 0) {
exit('parent process.');
}
// 將當前子進程提升會會話組組長 這是至關重要的一步
if (!posix_setsid()) {
exit('set sid error.');
}
// 二次fork
$pid = pcntl_fork();
if ($pid < 0) {
exit('fork error.');
} else if ($pid > 0) {
exit('parent process.');
} // 真正的邏輯代碼們 下面僅僅寫個循環以示例
for ($i = 1; $i <= 100; $i++) {
sleep(1);
file_put_contents('daemon.log', $i, FILE_APPEND);
}
?>
另外還有一種運行腳本的方式也值得一提:利用 screen / tmux 等軟件,將腳本運行在可以在一個虛擬終端之上。