linux下我们可以调用fork函数创建子进程,创建的子进程将会得到父进程的数据空间、堆、栈......副本(采用写时复制机制),子进程将会继承父进程的信号掩码、信号处理方式、当前工作目录、会话id、组id......。当子进程退出时父进程应当及时获取子进程退出状态,否则,如果父进程是一直在运行,那么子进程的退出状态将一直保存在内存中,直到父进程退出才释放。
我们可以使用如下几种方法避免僵尸进程的产生:
1.在fork后调用wait/waitpid函数取得子进程退出状态。
2.调用fork两次(第一次调用产生一个子进程,第二次调用fork是在第一个子进程中调用,同时将父进程退出(第一个子进程退出),此时的第二个子进程的父进程id为init进程id(注意:新版本Ubuntu并不是init的进程id))。
3.在程序中显示忽略SIGCHLD信号(子进程退出时会产生一个SIGCHLD信号,我们显示忽略此信号即可)。
4.捕获SIGCHLD信号并在捕获程序中调用wait/waitpid函数。
方法一:
- #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;
- }
复制代码
方法二:
- #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;
- }
- [b]方法三:[/b]
- #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;
- }
复制代码
方法四:
- #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");
- }
- }
复制代码 |