linux下我们可以调用fork函数创建子进程,创建的子进程将会得到父进程的数据空间、堆、栈......副本(采用写时复制机制),子进程将会继承父进程的信号掩码、信号处理方式、当前工作目录、会话id、组id......。当子进程退出时父进程应当及时获取子进程退出状态,否则,如果父进程是一直在运行,那么子进程的退出状态将一直保存在内存中,直到父进程退出才释放。
我们可以使用如下几种方法避免僵尸进程的产生:
1.在fork后调用wait/waitpid函数取得子进程退出状态。
2.调用fork两次(第一次调用产生一个子进程,第二次调用fork是在第一个子进程中调用,同时将父进程退出(第一个子进程退出),此时的第二个子进程的父进程id为init进程id(注意:新版本Ubuntu并不是init的进程id))。
3.在程序中显示忽略SIGCHLD信号(子进程退出时会产生一个SIGCHLD信号,我们显示忽略此信号即可)。
4.捕获SIGCHLD信号并在捕获程序中调用wait/waitpid函数。
方法一:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
#include "../common/common.h" int main( void ) { pid_t pid; if ((pid = fork()) < 0) { perror ( "fork error" ); return EXIT_FAILURE; } else if (0 == pid) { printf ( "[%ld] child process is running...\n" , ( long )getpid()); _exit(0); } //sleep(15); if (waitpid(pid, NULL, 0) < 0) { perror ( "waitpid error" ); return EXIT_FAILURE; } for (; ;) { pause(); } return EXIT_SUCCESS; } |
方法二:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
|
#include <sys/wait.h> #include "../common/common.h" int main( void ) { pid_t pid; if ((pid = fork()) < 0) { perror ( "fork error" ); return EXIT_FAILURE; } else if (0 == pid) { printf ( "first child is running..\n" ); /**在第一个子进程中再次fork***/ if ((pid = fork()) < 0) { perror ( "fork error" ); return EXIT_FAILURE; } else if (pid > 0) { /**父进程退出**/ printf ( "[%ld] first child is exit...\n" , ( long )getpid()); _exit(0); } sleep(2); /**确保父进程先运行**/ printf ( "second process pid: %ld, second process's parent pid: %ld\n" , ( long )getpid(), ( long )getppid()); //sleep(15); printf ( "[%ld] is exit..\n" , ( long )getpid()); _exit(0); } /***获得第一个子进程的退出状态***/ if (waitpid(pid, NULL, 0) < 0) { perror ( "waitpid error" ); return EXIT_FAILURE; } for (;;) pause(); return EXIT_SUCCESS; } |
方法三:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
#include <signal.h> #include "../common/common.h" int main( void ) { /***显示忽略SIGCHLD信号****/ if ( signal (SIGCHLD, SIG_IGN) == SIG_ERR) { perror ( "signal error" ); return EXIT_SUCCESS; } pid_t pid; int i; /**产生10个子进程***/ for (i=0; i<10; ++i) { if ((pid = fork()) < 0) { perror ( "fork error" ); return EXIT_FAILURE; } else if (0 == pid) { _exit(0); } sleep(2); continue ; } for (; ;) pause(); return EXIT_SUCCESS; } |
方法四:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
|
#include <signal.h> #include <sys/wait.h> #include "../common/common.h" void sig_chld( int signo); int main( void ) { /**捕获此信号, 此刻系统会立刻检测是否有次信号产生**/ if ( signal (SIGCHLD, sig_chld) == SIG_ERR) { handler_err( "signal error to SIGCHLD" ); } pid_t pid; int i; for (i=0; i<10; i++) { if ((pid = fork()) < 0) { handler_err( "fork error" ); } else if (0 == pid) { printf ( "child pid: %d\n" , getpid()); _exit(0); } sleep(1); continue ; } for (; ;) { pause(); } return EXIT_SUCCESS; } /**捕获到信号后会立刻执行此段代码***/ void sig_chld( int signo) { printf ( "receive child signal\n" ); if (waitpid(-1, NULL, 0) < 0) { perror ( "waitpid error" ); } if ( signal (SIGCHLD, sig_chld) == SIG_ERR) { perror ( "signal error to SIGCHLD" ); } } |
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。
原文链接:http://blog.csdn.net/ComingFlying/article/details/76718809