linux僵尸進程產生的原因以及如何避免產生僵尸進程defunct

給進程設置僵尸狀態的目的是維護子進程的信息,以便父進程在以后某個時間獲取。這些信息包括子進程的進程ID、終止狀態以及資源利用信息(CPU時間,內存使用量等等)。如果一個進程終止,而該進程有子進程處于僵尸狀態,那么它的所有僵尸子進程的父進程ID將被重置為1(init進程)。繼承這些子進程的init進程將清理它們(init進程將wait它們,從而去除僵尸狀態)。

??????? 但通常情況下,我們是不愿意留存僵尸進程的,它們占用內核中的空間,最終可能導致我們耗盡進程資源。那么為什么會產生僵尸進程以及如何避免產生僵尸進程呢?下邊我將從這兩個方面進行分析。

??? 僵尸進程的原因

??????? 我們知道,要在當前進程中生成一個子進程,一般需要調用fork這個系統調用,fork這個函數的特別之處在于一次調用,兩次返回,一次返回到父進程中,一次返回到子進程中,我們可以通過返回值來判斷其返回點:

復制代碼

pid_t child = fork();
if( child < 0  ) {     //fork error.perror("fork process fail.\n");
} else if( child ==0  ) {   // in child processprintf(" fork succ, this run in child process\n ");
} else {                        // in parent processprintf(" this run in parent process\n ");
}

復制代碼

??????? 如果子進程先于父進程退出, 同時父進程又沒有調用wait/waitpid,則該子進程將成為僵尸進程。通過ps命令,我們可以看到該進程的狀態為Z(表示僵死),如圖1所示:

??????? ddd1

?????????????????????????????????????????????????? (圖1)

??? 備注: 有些unix系統在ps命令輸出的COMMAND欄以<defunct>指明僵尸進程。

??????? 代碼如下:

復制代碼

if( child == -1 ) { //errorperror("\nfork child error.");exit(0);
} else if(child == 0){cout << "\nIm in child process:" <<  getpid() << endl;exit(0);
} else {cout << "\nIm in parent process."  << endl;sleep(600);
}

復制代碼

??????? 讓父進程休眠600s, 然后子進程先退出,我們就可以看到先退出的子進程成為僵尸進程了(進程狀態為Z)

?? 避免產生僵尸進程

??????? 我們知道了僵尸進程產生的原因,下邊我們看看如何避免產生僵尸進程。

??????? 一般,為了防止產生僵尸進程,在fork子進程之后我們都要wait它們;同時,當子進程退出的時候,內核都會給父進程一個SIGCHLD信號,所以我們可以建立一個捕獲SIGCHLD信號的信號處理函數,在函數體中調用wait(或waitpid),就可以清理退出的子進程以達到防止僵尸進程的目的。如下代碼所示:

復制代碼

void sig_chld( int signo ) {pid_t pid;int stat;pid = wait(&stat);    printf( "child %d exit\n", pid );return;
}int main() {signal(SIGCHLD,  &sig_chld);
}

復制代碼

??????? 現在main函數中給SIGCHLD信號注冊一個信號處理函數(sig_chld),然后在子進程退出的時候,內核遞交一個SIGCHLD的時候就會被主進程捕獲而進入信號處理函數sig_chld,然后再在sig_chld中調用wait,就可以清理退出的子進程。這樣退出的子進程就不會成為僵尸進程。

??????? 然后,即便我們捕獲SIGCHLD信號并且調用wait來清理退出的進程,仍然不能徹底避免產生僵尸進程;我們來看一種特殊的情況:

??????? 我們假設有一個client/server的程序,對于每一個連接過來的client,server都啟動一個新的進程去處理來自這個client的請求。然后我們有一個client進程,在這個進程內,發起了多個到server的請求(假設5個),則server會fork 5個子進程來讀取client輸入并處理(同時,當客戶端關閉套接字的時候,每個子進程都退出);當我們終止這個client進程的時候 ,內核將自動關閉所有由這個client進程打開的套接字,那么由這個client進程發起的5個連接基本在同一時刻終止。這就引發了5個FIN,每個連接一個。server端接受到這5個FIN的時候,5個子進程基本在同一時刻終止。這就又導致差不多在同一時刻遞交5個SIGCHLD信號給父進程,如圖2所示:

??????? tcp1111

?????????????????????????? (圖2)

??????? 正是這種同一信號多個實例的遞交造成了我們即將查看的問題。

??????? 我們首先運行服務器程序,然后運行客戶端程序,運用ps命令看以看到服務器fork了5個子進程,如圖3:

??????? t1

?????????????????????????????????? (圖3)

??????? 然后我們Ctrl+C終止客戶端進程,在我機器上邊測試,可以看到信號處理函數運行了3次,還剩下2個僵尸進程,如圖4:

??????? t2

????????????????????????????????? (圖4)

?????? 通過上邊這個實驗我們可以看出,建立信號處理函數并在其中調用wait并不足以防止出現僵尸進程,其原因在于:所有5個信號都在信號處理函數執行之前產生,而信號處理函數只執行一次,因為Unix信號一般是不排隊的(我的這篇博客中有提到http://www.cnblogs.com/yuxingfirst/p/3160697.html)。 更為嚴重的是,本問題是不確定的,依賴于客戶FIN到達服務器主機的時機,信號處理函數執行的次數并不確定。

?????? 正確的解決辦法是調用waitpid而不是wait,這個辦法的方法為:信號處理函數中,在一個循環內調用waitpid,以獲取所有已終止子進程的狀態。我們必須指定WNOHANG選項,他告知waitpid在有尚未終止的子進程在運行時不要阻塞。(我們不能在循環內調用wait,因為沒有辦法防止wait在尚有未終止的子進程在運行時阻塞,wait將會阻塞到現有的子進程中第一個終止為止),下邊的程序分別給出了這兩種處理辦法(func_wait, func_waitpid)。

復制代碼

//server.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <errno.h>
#include <error.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <string.h>
#include <signal.h>
#include <sys/wait.h>typedef void sigfunc(int);void func_wait(int signo) {pid_t pid;int stat;pid = wait(&stat);    printf( "child %d exit\n", pid );return;
}
void func_waitpid(int signo) {pid_t pid;int stat;while( (pid = waitpid(-1, &stat, WNOHANG)) > 0 ) {printf( "child %d exit\n", pid );}return;
}sigfunc* signal( int signo, sigfunc *func ) {struct sigaction act, oact;act.sa_handler = func;sigemptyset(&act.sa_mask);act.sa_flags = 0;if ( signo == SIGALRM ) {
#ifdef            SA_INTERRUPTact.sa_flags |= SA_INTERRUPT;    /* SunOS 4.x */
#endif} else {
#ifdef           SA_RESTARTact.sa_flags |= SA_RESTART;    /* SVR4, 4.4BSD */
#endif}if ( sigaction(signo, &act, &oact) < 0 ) {return SIG_ERR;}return oact.sa_handler;
} void str_echo( int cfd ) {ssize_t n;char buf[1024];
again:memset(buf, 0, sizeof(buf));while( (n = read(cfd, buf, 1024)) > 0 ) {write(cfd, buf, n); }if( n <0 && errno == EINTR ) {goto again; } else {printf("str_echo: read error\n");}
}int main() {signal(SIGCHLD, &func_waitpid);    int s, c;pid_t child;if( (s = socket(AF_INET, SOCK_STREAM, 0)) < 0 ) {int e = errno; perror("create socket fail.\n");exit(0);}struct sockaddr_in server_addr, child_addr; bzero(&server_addr, sizeof(server_addr));server_addr.sin_family = AF_INET;server_addr.sin_port = htons(9998);server_addr.sin_addr.s_addr = htonl(INADDR_ANY);if( bind(s, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0 ) {int e = errno; perror("bind address fail.\n");exit(0);}if( listen(s, 1024) < 0 ) {int e = errno; perror("listen fail.\n");exit(0);}while(1) {socklen_t chilen = sizeof(child_addr); if ( (c = accept(s, (struct sockaddr *)&child_addr, &chilen)) < 0 ) {perror("listen fail.");exit(0);}if( (child = fork()) == 0 ) {close(s); str_echo(c);exit(0);}close(c);}
}//client.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <errno.h>
#include <error.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <string.h>
#include <signal.h>void str_cli(FILE *fp, int sfd ) {char sendline[1024], recvline[2014];memset(recvline, 0, sizeof(sendline));memset(sendline, 0, sizeof(recvline));while( fgets(sendline, 1024, fp) != NULL ) {write(sfd, sendline, strlen(sendline)); if( read(sfd, recvline, 1024) == 0 ) {printf("server term prematurely.\n"); }fputs(recvline, stdout);memset(recvline, 0, sizeof(sendline));memset(sendline, 0, sizeof(recvline));}
}int main() {int s[5]; for (int i=0; i<5; i++) {if( (s[i] = socket(AF_INET, SOCK_STREAM, 0)) < 0 ) {int e = errno; perror("create socket fail.\n");exit(0);}}for (int i=0; i<5; i++) {struct sockaddr_in server_addr, child_addr; bzero(&server_addr, sizeof(server_addr));server_addr.sin_family = AF_INET;server_addr.sin_port = htons(9998);inet_pton(AF_INET, "127.0.0.1", &server_addr.sin_addr);if( connect(s[i], (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0 ) {perror("connect fail."); exit(0);}}sleep(10);str_cli(stdin, s[0]);exit(0);
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/384976.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/384976.shtml
英文地址,請注明出處:http://en.pswp.cn/news/384976.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

linux下僵尸進程(Defunct進程)的產生與避免

在測試基于 DirectFBGstreamer 的視頻聯播系統的一個 Demo 的時候&#xff0c;其中大量使用 system 調用的語句&#xff0c;例如在 menu 代碼中的 system("./play") &#xff0c;而且多次執行&#xff0c;這種情況下&#xff0c;在 ps -ef 列表中出現了大量的 defunc…

文件操作函數

fopen()函數參數&#xff1a; r 只讀的方式打開文件。 打開成功返回文件指針&#xff0c; 打開失敗返回NULL r 以讀寫方式打開文件。 文件必須存在 rb 以二進制模式讀寫文件&#xff0c;文件必須存在 rw 讀寫一個二進制文件&#xff0c;允許讀和寫 w 打開只寫文件&…

讀過的最好的epoll講解

首先我們來定義流的概念&#xff0c;一個流可以是文件&#xff0c;socket&#xff0c;pipe等等可以進行I/O操作的內核對象。 不管是文件&#xff0c;還是套接字&#xff0c;還是管道&#xff0c;我們都可以把他們看作流。 之后我們來討論I/O的操作&#xff0c;通過read&#xf…

文件操作函數(讀寫)

文件文本排序&#xff1a; 數組冒泡&#xff1a; #include<stdio.h>void swap(int *a,int *b) {int temp *a;*a *b;*b temp; }void bubble(int *p,int n) {int i;int j;for(i 0; i < n; i){for(j 1; j < n - i; j){if(p[j - 1] > p[j]){swap(&p[j-1],&…

文件操作(升級)

計算字符串“25 32 ” #include<stdio.h> #include<string.h>int calc_string(char *s) {char buf1[100] {0};char oper 0;char buf2[100] {0};int len strlen(s);int i;for(i 0; i < len; i){if( s[i] || - s[i] || * s[i] || / s[i] ){strncpy…

C語言指針轉換為intptr_t類型

C語言指針轉換為intptr_t類型 1、前言 今天在看代碼時&#xff0c;發現將之一個指針賦值給一個intptr_t類型的變量。由于之前沒有見過intptr_t這樣數據類型&#xff0c;憑感覺認為intptr_t是int類型的指針。感覺很奇怪&#xff0c;為何要將一個指針這樣做呢&#xff1f;如是果…

nginx epoll詳解

nginx epoll 事件模型 nginx做為一個異步高效的事件驅動型web服務器&#xff0c;在linux平臺中當系統支持epoll時nginx默認采用epoll來高效的處理事件。nginx中使用ngx_event_t結構來表示一個事件&#xff0c;先介紹下ngx_event_t結構體中成員的含義&#xff1a; struct ngx_ev…

Inotify機制

描述 Inotify API用于檢測文件系統變化的機制。Inotify可用于檢測單個文件&#xff0c;也可以檢測整個目錄。當檢測的對象是一個目錄的時候&#xff0c;目錄本身和目錄里的內容都會成為檢測的對象。 此種機制的出現的目的是當內核空間發生某種事件之后&#xff0c;可以立即通…

文件操作(二進制文件加密解密)

加密 #include<stdio.h> #include<string.h>void code(char *p,size_t n) {size_t i;for(i 0; i < n; i){p[i] 3;} }int main() {FILE *p1 fopen("./a.txt","r");FILE *p2 fopen("./b.txt","w");char buf[1024] {…

北京加密機現場select問題

問題描述 北京項目通過調用我們提供的庫libsigxt.a與加密機通信&#xff0c;c/s架構&#xff0c;客戶端啟用多個線程&#xff0c;每個線程流程有以下三步&#xff0c;連接加密機&#xff0c;簽名&#xff0c;關閉鏈接。在正常運行一段時間后會出現不能連接加密機服務問題。 連…

拼接字符串(帶參程序)

1.用strcat拼接函數可以實現 #include<stdio.h> #include<string.h>int main(int argc,char ** argv) {char str[100] {0};int i;for( i 1; i < argc; i){strcat(str,argv[i]);}printf("str %s\n",str);return 0; } 2.用sprintf函數也可以實現 #in…

詳細解釋signal和sigaction以及SIG_BLOCK

signal&#xff0c;此函數相對簡單一些&#xff0c;給定一個信號&#xff0c;給出信號處理函數則可&#xff0c;當然&#xff0c;函數簡單&#xff0c;其功能也相對簡單許多&#xff0c;簡單給出個函數例子如下&#xff1a; [cpp] view plain copy 1 #include <signal.h>…

處理SIGCHLD信號

在上一講中&#xff0c;我們使用fork函數得到了一個簡單的并發服務器。然而&#xff0c;這樣的程序有一個問題&#xff0c;就是當子進程終止時&#xff0c;會向父進程發送一個SIGCHLD信號&#xff0c;父進程默認忽略&#xff0c;導致子進程變成一個僵尸進程。僵尸進程一定要處理…

文件操作(stat)

/*** stat.c ***/ #include<stdio.h> #include<string.h> #include<sys/stat.h> #include<stdlib.h>int main() {struct stat st {0}; //定義一個結構體&#xff0c;名字叫ststat("./a.txt",&st); //調用完stat函數之后&…

nginx源碼閱讀(一).綜述

前言 nginx作為一款開源的輕量級高性能web服務器,是非常值得立志從事服務端開發方向的人學習的。現今nginx的最新版本是nginx-1.13.6,代碼量也日漸龐大,但是由于其核心思想并沒改變,為了降低閱讀難度,我選擇的是nginx-1.0.15版本,并且由于時間和水平有限,重點關注的是nginx的啟…

文件操作(stat函數)

stat函數可以獲取文件信息 /*** stat.c ***/ #include<stdio.h> #include<string.h> #include<sys/stat.h> #include<stdlib.h>int main() {struct stat st {0}; //定義一個結構體&#xff0c;名字叫ststat("./a.txt",&st); …

文件操作(結構體)

將結構體內容寫入到文件中 #include<stdio.h> #include<string.h>struct student {char name[100];int age; };int main() {struct student st {"wangqinghe",30};FILE * p fopen("./c.txt","wb");fwrite(&st,sizeof(st),1,p…

nginx源碼閱讀(二).初始化:main函數及ngx_init_cycle函數

前言 在分析源碼時,我們可以先把握主干,然后其他部分再挨個分析就行了。接下來我們先看看nginx的main函數干了些什么。 main函數 這里先介紹一些下面會遇到的變量類型: ngx_int_t: typedef intptr_t ngx_int_t; 64位機器上,intptr_t為long int, 即typedef long int intptr_…

EAGAIN、EWOULDBLOCK、EINTR與非阻塞

EWOULDBLOCK&#xff1a;用于非阻塞模式&#xff0c;不需要重新讀或者寫 EINTR&#xff1a;指操作被中斷喚醒&#xff0c;需要重新讀/寫   在Linux環境下開發經常會碰到很多錯誤(設置errno)&#xff0c;其中EAGAIN是其中比較常見的一個錯誤(比如用在非阻塞操作中)。 從字面上…

文件操作(排序)

文本文件&#xff0c;每行代表一個整數&#xff0c;范圍在0~512之間&#xff1b; 要求&#xff1a;對文件排序&#xff0c;不使用堆空間&#xff0c;只使用棧空間。 用srand()和rand()函數生成一定量的隨機數 /*** file.c ***/ #include<stdio.h> #include<string.h&g…