今天在維護伺服器的時候,發現有5個nova-novncproxy的殭屍行程。
26327 ? S 0:05 \_ /usr/bin/python /usr/bin/nova-novncproxy --config-file=/etc/nova/nova.conf 4765 ? Z 0:00 \_ [nova-novncproxy] <defunct> 4766 ? Z 0:00 \_ [nova-novncproxy] <defunct> 4767 ? Z 0:00 \_ [nova-novncproxy] <defunct> 4768 ? Z 0:00 \_ [nova-novncproxy] <defunct> 4769 ? Z 0:00 \_ [nova-novncproxy] <defunct>
之前對殭屍進程的了解並不深,趕緊找了篇相關文章來學習一下,該如何處理。
定義
In UNIX System terminology, a process that has terminated,but whose parent has not yet waited for it, is called a zombie.
(呼叫wait / waitpid)他, 那麼他將變成一個殭屍進程. 在fork()/execve()過程中,假設子進程結束時父進程仍存在,而父進程fork()之前既沒安裝SIGCHLD信號處理函數呼叫waitpid()等待子程序結束,又沒有明確忽略該訊號,子程序成為殭屍程序。
如何查看linux系統上的殭屍進程,如何統計有多少殭屍進程?
#ps -ef | grep defunct
或找出狀態為Z的進程,Z就是代表zombie process,殭屍進程的意思。
另外使用top指令查看時有一欄為S,如果狀態為Z說明它就是殭屍行程。
Tasks: 95 total, 1 running, 94 sleeping, 0 stopped, 0 zombie
top指令中也統計了殭屍進程。或使用下面的命令:
ps -ef | grep defunct | grep -v grep | wc -l
如何殺死殭屍進程?
一般殭屍行程很難直接kill掉,不過您可以kill殭屍爸爸。父進程死後,殭屍進程成為”孤兒進程”,過繼給1號進程init,init始終會負責清理殭屍進程.它產生的所有殭屍進程也跟著消失。
ps -e -o ppid,stat | grep Z | cut -d” ” -f2 | xargs kill -9
或
,p ^[Zz ]' | awk '{print $2}'`當然您可以自己編寫更好的shell腳本,歡迎與大家分享。 我將nova-novncproxy stop後再start,殭屍進程即消失,問題解決。 另外子進程死後,會發送SIGCHLD訊號給父進程,父進程收到此訊號後,執行waitpid()函數為子進程收屍。就是基於這樣的原理:就算父進程沒有呼叫wait,核心也會向它發送SIGCHLD訊息,而此時,儘管對它的預設處理是忽略,如果想回應這個訊息,可以設定一個處理函數。 如何避免殭屍進程呢? 處理SIGCHLD訊號並不是必須的。但對於某些進程,特別是伺服器進程往往在請求到來時產生子進程處理請求。如果父行程不等待子行程結束,子行程將成為殭屍行程(zombie)從而佔用系統資源。如果父進程等待子進程結束,將增加父進程的負擔,影響伺服器進程的並發效能。在Linux下 可以簡單地將 SIGCHLD訊號的操作設為SIG_IGN。signal(SIGCHLD,SIG_IGN);
這樣,核心在子行程結束時不會產生殭屍行程。這點與BSD4不同,BSD4下必須顯式等待子進程結束才能釋放殭屍進程