Linux守護(hù)進(jìn)程的啟動方法
"守護(hù)進(jìn)程"(daemon)就是一直在后臺運行的進(jìn)程(daemon)。
一、問題的由來
Web應(yīng)用寫好后,下一件事就是啟動,讓它一直在后臺運行。
這并不容易。舉例來說,下面是一個最簡單的Node應(yīng)用server.js,只有6行。
var http = require('http');
http.createServer(function(req, res) {
res.writeHead(200, {'Content-Type': 'text/plain'});
res.end('Hello World');
}).listen(5000);
你在命令行下啟動它。
$ node server.js
看上去一切正常,所有人都能快樂地訪問 5000 端口了。但是,一旦你退出命令行窗口,這個應(yīng)用就一起退出了,無法訪問了。
怎么才能讓它變成系統(tǒng)的守護(hù)進(jìn)程(daemon),成為一種服務(wù)(service),一直在那里運行呢?
二、前臺任務(wù)與后臺任務(wù)
上面這樣啟動的腳本,稱為"前臺任務(wù)"(foreground job)。它會獨占命令行窗口,只有運行完了或者手動中止,才能執(zhí)行其他命令。
變成守護(hù)進(jìn)程的第一步,就是把它改成"后臺任務(wù)"(background job)。
$ node server.js &
只要在命令的尾部加上符號&,啟動的進(jìn)程就會成為"后臺任務(wù)"。如果要讓正在運行的"前臺任務(wù)"變?yōu)?后臺任務(wù)",可以先按ctrl + z,然后執(zhí)行bg命令(讓最近一個暫停的"后臺任務(wù)"繼續(xù)執(zhí)行)。
"后臺任務(wù)"有兩個特點。
- 繼承當(dāng)前 session (對話)的標(biāo)準(zhǔn)輸出(stdout)和標(biāo)準(zhǔn)錯誤(stderr)。因此,后臺任務(wù)的所有輸出依然會同步地在命令行下顯示。
- 不再繼承當(dāng)前 session 的標(biāo)準(zhǔn)輸入(stdin)。你無法向這個任務(wù)輸入指令了。如果它試圖讀取標(biāo)準(zhǔn)輸入,就會暫停執(zhí)行(halt)。
可以看到,"后臺任務(wù)"與"前臺任務(wù)"的本質(zhì)區(qū)別只有一個:是否繼承標(biāo)準(zhǔn)輸入。所以,執(zhí)行后臺任務(wù)的同時,用戶還可以輸入其他命令。
三、SIGHUP信號
變?yōu)?后臺任務(wù)"后,一個進(jìn)程是否就成為了守護(hù)進(jìn)程呢?或者說,用戶退出 session 以后,"后臺任務(wù)"是否還會繼續(xù)執(zhí)行?
Linux系統(tǒng)是這樣設(shè)計的。
- 用戶準(zhǔn)備退出 session
- 系統(tǒng)向該 session 發(fā)出SIGHUP信號
- session 將SIGHUP信號發(fā)給所有子進(jìn)程
- 子進(jìn)程收到SIGHUP信號后,自動退出
上面的流程解釋了,為什么"前臺任務(wù)"會隨著 session 的退出而退出:因為它收到了SIGHUP信號。
那么,"后臺任務(wù)"是否也會收到SIGHUP信號?
這由 Shell 的huponexit參數(shù)決定的。
$ shopt | grep huponexit
執(zhí)行上面的命令,就會看到huponexit參數(shù)的值。
大多數(shù)Linux系統(tǒng),這個參數(shù)默認(rèn)關(guān)閉(off)。因此,session 退出的時候,不會把SIGHUP信號發(fā)給"后臺任務(wù)"。所以,一般來說,"后臺任務(wù)"不會隨著 session 一起退出。
四、disown 命令
通過"后臺任務(wù)"啟動"守護(hù)進(jìn)程"并不保險,因為有的系統(tǒng)的huponexit參數(shù)可能是打開的(on)。
更保險的方法是使用disown命令。它可以將指定任務(wù)從"后臺任務(wù)"列表(jobs命令的返回結(jié)果)之中移除。一個"后臺任務(wù)"只要不在這個列表之中,session 就肯定不會向它發(fā)出SIGHUP信號。
$ node server.js & $ disown
執(zhí)行上面的命令以后,server.js進(jìn)程就被移出了"后臺任務(wù)"列表。你可以執(zhí)行jobs命令驗證,輸出結(jié)果里面,不會有這個進(jìn)程。
disown的用法如下。
# 移出最近一個正在執(zhí)行的后臺任務(wù) $ disown # 移出所有正在執(zhí)行的后臺任務(wù) $ disown -r # 移出所有后臺任務(wù) $ disown -a # 不移出后臺任務(wù),但是讓它們不會收到SIGHUP信號 $ disown -h # 根據(jù)jobId,移出指定的后臺任務(wù) $ disown %2 $ disown -h %2
五、標(biāo)準(zhǔn) I/O
使用disown命令之后,還有一個問題。那就是,退出 session 以后,如果后臺進(jìn)程與標(biāo)準(zhǔn)I/O有交互,它還是會掛掉。
還是以上面的腳本為例,現(xiàn)在加入一行。
var http = require('http');
http.createServer(function(req, res) {
console.log('server starts...'); // 加入此行
res.writeHead(200, {'Content-Type': 'text/plain'});
res.end('Hello World');
}).listen(5000);
啟動上面的腳本,然后再執(zhí)行disown命令。
$ node server.js & $ disown
接著,你退出 session,訪問5000端口,就會發(fā)現(xiàn)連不上。
這是因為"后臺任務(wù)"的標(biāo)準(zhǔn) I/O 繼承自當(dāng)前 session,disown命令并沒有改變這一點。一旦"后臺任務(wù)"讀寫標(biāo)準(zhǔn) I/O,就會發(fā)現(xiàn)它已經(jīng)不存在了,所以就報錯終止執(zhí)行。
為了解決這個問題,需要對"后臺任務(wù)"的標(biāo)準(zhǔn) I/O 進(jìn)行重定向。
$ node server.js > stdout.txt 2> stderr.txt < /dev/null & $ disown
上面這樣執(zhí)行,基本上就沒有問題了。
六、nohup 命令
還有比disown更方便的命令,就是nohub。
$ nohup node server.js &
nohup命令對server.js進(jìn)程做了三件事。
- 阻止SIGHUP信號發(fā)到這個進(jìn)程。
- 關(guān)閉標(biāo)準(zhǔn)輸入。該進(jìn)程不再能夠接收任何輸入,即使運行在前臺。
- 重定向標(biāo)準(zhǔn)輸出和標(biāo)準(zhǔn)錯誤到文件nohup.out。
也就是說,nohup命令實際上將子進(jìn)程與它所在的 session 分離了。
注意,nohup命令不會自動把進(jìn)程變?yōu)?后臺任務(wù)",所以必須加上&符號。
七、Screen 命令與 Tmux 命令
另一種思路是使用 terminal multiplexer (終端復(fù)用器:在同一個終端里面,管理多個session),典型的就是 Screen 命令和 Tmux 命令。
它們可以在當(dāng)前 session 里面,新建另一個 session。這樣的話,當(dāng)前 session 一旦結(jié)束,不影響其他 session。而且,以后重新登錄,還可以再連上早先新建的 session。
Screen 的用法如下。
# 新建一個 session $ screen $ node server.js
然后,按下ctrl + A和ctrl + D,回到原來的 session,從那里退出登錄。下次登錄時,再切回去。
$ screen -r
如果新建多個后臺 session,就需要為它們指定名字。
$ screen -S name # 切回指定 session $ screen -r name $ screen -r pid_number # 列出所有 session $ screen -ls
如果要停掉某個 session,可以先切回它,然后按下ctrl + c和ctrl + d。
Tmux 比 Screen 功能更多、更強大,它的基本用法如下。
$ tmux $ node server.js # 返回原來的session $ tmux detach
除了tmux detach,另一種方法是按下Ctrl + B和d ,也可以回到原來的 session。
# 下次登錄時,返回后臺正在運行服務(wù)session $ tmux attach
如果新建多個 session,就需要為每個 session 指定名字。
# 新建 session $ tmux new -s session_name # 切換到指定 session $ tmux attach -t session_name # 列出所有 session $ tmux list-sessions # 退出當(dāng)前 session,返回前一個 session $ tmux detach # 殺死指定 session $ tmux kill-session -t session-name
八、Node 工具
對于 Node 應(yīng)用來說,可以不用上面的方法,有一些專門用來啟動的工具:forever,nodemon 和 pm2。
forever 的功能很簡單,就是保證進(jìn)程退出時,應(yīng)用會自動重啟。
# 作為前臺任務(wù)啟動 $ forever server.js # 作為服務(wù)進(jìn)程啟動 $ forever start app.js # 停止服務(wù)進(jìn)程 $ forever stop Id # 重啟服務(wù)進(jìn)程 $ forever restart Id # 監(jiān)視當(dāng)前目錄的文件變動,一有變動就重啟 $ forever -w server.js # -m 參數(shù)指定最多重啟次數(shù) $ forever -m 5 server.js # 列出所有進(jìn)程 $ forever list
nodemon一般只在開發(fā)時使用,它最大的長處在于 watch 功能,一旦文件發(fā)生變化,就自動重啟進(jìn)程。
# 默認(rèn)監(jiān)視當(dāng)前目錄的文件變化 $ nodemon server.js # 監(jiān)視指定文件的變化 $ nodemon --watch app --watch libs server.js
pm2 的功能最強大,除了重啟進(jìn)程以外,還能實時收集日志和監(jiān)控。
# 啟動應(yīng)用 $ pm2 start app.js # 指定同時起多少個進(jìn)程(由CPU核心數(shù)決定),組成一個集群 $ pm2 start app.js -i max # 列出所有任務(wù) $ pm2 list # 停止指定任務(wù) $ pm2 stop 0 # 重啟指定任務(wù) $ pm2 restart 0 # 刪除指定任務(wù) $ pm2 delete 0 # 保存當(dāng)前的所有任務(wù),以后可以恢復(fù) $ pm2 save # 列出每個進(jìn)程的統(tǒng)計數(shù)據(jù) $ pm2 monit # 查看所有日志 $ pm2 logs # 導(dǎo)出數(shù)據(jù) $ pm2 dump # 重啟所有進(jìn)程 $ pm2 kill $ pm2 resurect # 啟動web界面 http://localhost:9615 $ pm2 web
十、Systemd
除了專用工具以外,Linux系統(tǒng)有自己的守護(hù)進(jìn)程管理工具 Systemd 。它是操作系統(tǒng)的一部分,直接與內(nèi)核交互,性能出色,功能極其強大。我們完全可以將程序交給 Systemd ,讓系統(tǒng)統(tǒng)一管理,成為真正意義上的系統(tǒng)服務(wù)。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助。
- linux shell實現(xiàn)守護(hù)進(jìn)程腳本
- php守護(hù)進(jìn)程 加linux命令nohup實現(xiàn)任務(wù)每秒執(zhí)行一次
- 詳解Linux中的守護(hù)進(jìn)程
- linux下如何創(chuàng)建守護(hù)進(jìn)程的步驟
- C語言編寫Linux守護(hù)進(jìn)程實例
- Python實現(xiàn)Linux下守護(hù)進(jìn)程的編寫方法
- Java實現(xiàn)Linux下雙守護(hù)進(jìn)程
- linux 守護(hù)進(jìn)程詳解及建立守護(hù)進(jìn)程
- linux下的守護(hù)進(jìn)程
- linux守護(hù)進(jìn)程服務(wù)daemon、nohup、systemd的區(qū)別
相關(guān)文章
Mac通過不同終端SSH連接遠(yuǎn)程服務(wù)器的講解
今天小編就為大家分享一篇關(guān)于Mac通過不同終端SSH連接遠(yuǎn)程服務(wù)器的講解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-03-03
CentOS 6.5 web服務(wù)器apache的安裝與基本設(shè)置
這篇文章主要介紹了CentOS 6.5 web服務(wù)器apache的安裝與基本設(shè)置,需要的朋友可以參考下2017-09-09
yum 安裝memcache permission denied(拒絕訪問) 問題
這篇文章主要介紹了yum 安裝memcache permission denied(拒絕訪問) 問題,需要的朋友可以參考下2017-03-03
CentOS 7.6安裝MySQL 5.7 GA版的教程圖解
本文通過圖文并茂的形式給大家介紹了CentOS 7.6安裝MySQL 5.7 GA版,需要的朋友可以參考下2019-09-09
虛擬機安裝centos7的坑之找不到網(wǎng)卡問題及解決
這篇文章主要介紹了虛擬機安裝centos7的坑之找不到網(wǎng)卡問題及解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-10-10
Centos7.4 zabbix3.4.7源碼安裝的方法步驟
這篇文章主要介紹了Centos7.4 zabbix3.4.7源碼安裝的方法步驟,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-06-06
Ubuntu22.04系統(tǒng):fatal:?無法連接到?github.com
這篇文章主要介紹了Ubuntu22.04系統(tǒng):fatal:?無法連接到?github.com的相關(guān)資料,需要的朋友可以參考下2024-03-03
centos7修改網(wǎng)卡后無法上網(wǎng)問題解決過程
大家好,本篇文章主要講的是centos7修改網(wǎng)卡后無法上網(wǎng)問題解決過程,感興趣的同學(xué)趕快來看一看吧,對你有幫助的話記得收藏一下,方便下次瀏覽2021-12-12

