How do you kill zombie process using wait()
Solution 1
How do you know (and) where to put the "wait()" statement to kill zombie processes?
If your parent spawns only a small, fixed number of children; does not care when or whether they stop, resume, or finish; and itself exits quickly, then you do not need to use wait()
or waitpid()
to clean up the child processes. The init process (pid 1) takes responsibility for orphaned child processes, and will clean them up when they finish.
Under any other circumstances, however, you must wait()
for child processes. Doing so frees up resources, ensures that the child has finished, and allows you to obtain the child's exit status. Via waitpid()
you can also be notified when a child is stopped or resumed by a signal, if you so wish.
As for where to perform the wait,
- You must ensure that only the parent
wait()
s. - You should wait at or before the earliest point where you need the child to have finished (but not before forking), OR
- if you don't care when or whether the child finishes, but you need to clean up resources, then you can periodically call
waitpid(-1, NULL, WNOHANG)
to collect a zombie child if there is one, without blocking if there isn't any.
In particular, you must not wait()
(unconditionally) immediately after fork()
ing because parent and child run the same code. You must use the return value of fork()
to determine whether you are in the child (return value == 0), or in the parent (any other return value). Furthermore, the parent must wait()
only if forking was successful, in which case fork()
returns the child's pid, which is always greater than zero. A return value less than zero indicates failure to fork.
Your program doesn't really need to wait()
because it spawns exactly four (not three) children, then exits. However, if you wanted the parent to have at most one live child at any time, then you could write it like this:
int main() {
pid_t child;
int i;
printf("-----------------------------------\n");
about("Parent");
for (i = 0; i < 3; i++) {
printf("Now .. Forking !!\n");
child = fork();
if (child < 0) {
perror ("Unable to fork");
break;
} else if (child == 0) {
printf ("In child #%d\n", (i+1));
about ("Child");
break;
} else {
/* in parent */
if (waitpid(child, NULL, 0) < 0) {
perror("Failed to collect child process");
break;
}
}
}
return 0;
}
If the parent exits before one or more of its children, which can happen if it does not wait, then the child will thereafter see its parent process being pid 1.
Others have already answered how to get a zombie process list via th ps
command. You may also be able to see zombies via top
. With your original code you are unlikely to catch a glimpse of zombies, however, because the parent process exits very quickly, and init
will then clean up the zombies it leaves behind.
Solution 2
How do you know (and) where to put the "wait()" statement to kill zombie processes?
You can use wait()
anywhere in the parent process, and when the child process terminates it'll be removed from the system. Where to put it is up to you, in your specific case you probably want to put it immediately after the child = fork();
line so that the parent process won't resume its execution until its child has exited.
What is the command to view zombie processes if you have Linux virtual box?
You can use the ps aux
command to view all processes in the system (including zombie processes), and the STAT
column will be equal to Z
if the process is a zombie. An example output would be:
USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND
daniel 1000 0.0 0.0 0 0 ?? Z 17:15 0:00 command
mudwomp
Updated on October 19, 2020Comments
-
mudwomp over 3 years
I have this code that requires a parent to fork 3 children.
How do you know (and) where to put the "wait()" statement to kill
zombie processes?What is the command to view zombie processes if you have Linux virtual box?
main(){ pid_t child; printf("-----------------------------------\n"); about("Parent"); printf("Now .. Forking !!\n"); child = fork(); int i=0; for (i=0; i<3; i++){ if (child < 0) { perror ("Unable to fork"); break; } else if (child == 0){ printf ("creating child #%d\n", (i+1)); about ("Child"); break; } else{ child = fork(); } } } void about(char * msg){ pid_t me; pid_t oldone; me = getpid(); oldone = getppid(); printf("***[%s] PID = %d PPID = %d.\n", msg, me, oldone); }
-
mudwomp about 9 yearsThank you for getting back to me. While testing this without a
wait()
, when spamming compile, I see that sometimes I get a 1 for PPID instead of the PID of the parent. With thewait(5)
inside the parent function, I do not see any. Does that mean having thewait(5)
inside the function is getting rid of zombie processes? -
mudwomp about 9 yearsAlso, when I type
ps aux
, I end up going to line(?)26433 tc
. It just spamstc
all the way down so I'm not sure where to look. -
Daniel Kleinstein about 9 yearsAssuming you have zombie processes, run
ps aux | grep Z
to view just the lines containingZ
(if your program has finished its execution then this won't show you anything relevant, as all the zombie children have been removed from the system). As for your first issue, what's happening is that your child process is running after the parent process has exited, so its parent is now theinit
process which haspid == 1
. By callingwait
you're preventing the parent process from exiting until at least one of its children has finished its execution.