Andere Prozesse

Bisher haben wir über Prozesse, deren Erstellung, übergeordnete und untergeordnete Prozesse usw. gesprochen. Die Diskussion wird unvollständig sein, ohne andere verwandte Prozesse wie den Orphan-Prozess, den Zombie-Prozess und den Daemon-Prozess zu diskutieren.

Orphan Process

Wie der Name schon sagt, impliziert Orphan einen Prozess ohne Eltern. Wenn wir ein Programm oder eine Anwendung ausführen, ist der übergeordnete Prozess für die Anwendung Shell. Wenn wir einen Prozess mit fork () erstellen, ist der neu erstellte Prozess der untergeordnete Prozess und der Prozess, der das untergeordnete Prozess erstellt hat, der übergeordnete Prozess. Der übergeordnete Prozess hierfür ist wiederum die Shell. Das übergeordnete Element aller Prozesse ist natürlich der Init-Prozess (Prozess-ID → 1).

Das Obige ist jedoch ein übliches Szenario. Was passiert, wenn der übergeordnete Prozess vor dem untergeordneten Prozess beendet wird? Das Ergebnis ist, dass der untergeordnete Prozess jetzt zum verwaisten Prozess wird. Was ist dann mit dem übergeordneten Element? Das neue übergeordnete Element ist das übergeordnete Element aller Prozesse. Dies ist nichts anderes als der Init-Prozess (Prozess-ID - 1).

Versuchen wir dies anhand des folgenden Beispiels zu verstehen.

/ * Dateiname: orphan_process.c * /

#include<stdio.h>
#include<stdlib.h>

int main() {
   int pid;
   system("ps -f");
   pid = fork();
   if (pid == 0) {
      printf("Child: pid is %d and ppid is %d\n",getpid(),getppid());
      sleep(5);
      printf("Child: pid is %d and ppid is %d\n",getpid(),getppid());
      system("ps -f");
   } else {
      printf("Parent: pid is %d and ppid is %d\n",getpid(),getppid());
      sleep(2);
      exit(0);
   }
   return 0;
}

Kompilierungs- und Ausführungsschritte

UID         PID   PPID  C STIME TTY    TIME CMD
4581875  180558      0  0 09:19  ?     00:00:00 sh -c cd /home/cg/root/4581875; 
                                       timeout 10s main
4581875  180564 180558  0 09:19  ?     00:00:00 timeout 10s main
4581875  180565 180564  0 09:19  ?     00:00:00 main
4581875  180566 180565  0 09:19  ?     00:00:00 ps -f
Parent: pid is 180565 and ppid is 180564
UID         PID   PPID  C STIME TTY    TIME CMD
4581875  180567      0  0 09:19  ?     00:00:00 main
4581875  180820 180567  0 09:19  ?     00:00:00 ps -f
Child: pid is 180567 and ppid is 180565
Child: pid is 180567 and ppid is 0

Zombie-Prozess

Nehmen wir in einfachen Worten an, Sie haben zwei Prozesse, nämlich den übergeordneten und den untergeordneten Prozess. Es liegt in der Verantwortung des übergeordneten Prozesses, auf den untergeordneten Prozess zu warten und dann den untergeordneten Prozesseintrag aus der Prozesstabelle zu bereinigen. Was ist, wenn der übergeordnete Prozess nicht bereit ist, auf den untergeordneten Prozess zu warten, und der untergeordnete Prozess in der Zwischenzeit seine Arbeit erledigt und beendet wird? Jetzt würde der untergeordnete Prozess zum Zombie-Prozess. Natürlich wird der Zombie-Prozess bereinigt, nachdem der übergeordnete Prozess fertig ist.

Lassen Sie uns dies anhand eines Beispiels verstehen.

/ * Dateiname: zombie_process.c * /

#include<stdio.h>
#include<stdlib.h>

int main() {
   int pid;
   pid = fork();
   if (pid == 0) {
      system("ps -f");
      printf("Child: pid is %d and ppid is %d\n",getpid(),getppid());
      exit(0);
   } else {
      printf("Parent: pid is %d and ppid is %d\n",getpid(),getppid());
      sleep(10);
      system("ps aux|grep Z");
   }
   return 0;
}

Kompilierungs- und Ausführungsschritte

UID         PID   PPID  C STIME TTY    TIME CMD
4581875  184946      0  0 09:20  ?     00:00:00 sh -c cd /home/cg/root/4581875; 
                                       timeout 10s main
4581875  184952 184946  0 09:20  ?     00:00:00 timeout 10s main
4581875  184953 184952  0 09:20  ?     00:00:00 main
4581875  184954 184953  0 09:20  ?     00:00:00 main
4581875  184955 184954  0 09:20  ?     00:00:00 ps -f
Child: pid is 184954 and ppid is 184953

Daemon-Prozess

In einfachen Worten wird der Prozess, dem keine Shell oder kein Terminal zugeordnet ist, als Daemon-Prozess bezeichnet. Warum wird das benötigt? Dies sind die Prozesse, die im Hintergrund ausgeführt werden, um Aktionen in vordefinierten Intervallen auszuführen und auch auf bestimmte Ereignisse zu reagieren. Der Daemon-Prozess sollte keine Benutzerinteraktion haben, da er als Hintergrundprozess ausgeführt wird.

Die internen Linux-Daemon-Prozesse enden normalerweise mit dem Buchstaben "d", z. B. Kernel-Daemons (ksoftirqd, kblockd, kswapd usw.), Printing Daemons (cupsd, lpd usw.), File Service Daemons (smbd, nmbd usw.). , Daemons für administrative Datenbanken (ypbind, ypserv usw.), Daemons für E-Mail (sendmail, popd, smtpd usw.), Daemons für Remote-Anmeldung und Befehlsausführung (sshd, in.telnetd usw.), Daemons für Booten und Konfiguration (dhcpd) , udevd usw.), init-Prozess (init), cron-Daemon, atd-Daemon usw.

Lassen Sie uns nun sehen, wie Sie einen Daemon-Prozess erstellen. Es folgen die Schritte -

Step 1- Erstellen Sie einen untergeordneten Prozess. Jetzt haben wir zwei Prozesse - den übergeordneten Prozess und den untergeordneten Prozess

Normalerweise lautet die Prozesshierarchie SHELL → PARENT PROCESS → CHILD PROCESS

Step 2- Beenden Sie den übergeordneten Prozess durch Beenden. Der untergeordnete Prozess wird nun zum verwaisten Prozess und wird vom Init-Prozess übernommen.

Die Hierarchie lautet nun INIT PROCESS → CHILD PROCESS

Step 3- Durch Aufrufen des Systemaufrufs setsid () wird eine neue Sitzung erstellt, wenn der aufrufende Prozess kein Prozessgruppenleiter ist. Jetzt wird der aufrufende Prozess zum Gruppenleiter der neuen Sitzung. Dieser Prozess ist der einzige Prozess in dieser neuen Prozessgruppe und in dieser neuen Sitzung.

Step 4 - Setzen Sie die Prozessgruppen-ID und die Sitzungs-ID auf die PID des aufrufenden Prozesses.

Step 5 - Schließen Sie die Standard-Dateideskriptoren (Standardeingabe, Standardausgabe und Standardfehler) des Prozesses, da Terminal und Shell jetzt von der Anwendung getrennt sind.

/ * Dateiname: daemon_test.c * /

#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<unistd.h>
#include<fcntl.h>
#include<stdlib.h>
#include<string.h>

int main(int argc, char *argv[]) {
   pid_t pid;
   int counter;
   int fd;
   int max_iterations;
   char buffer[100];
   if (argc < 2)
   max_iterations = 5;
   else {
      max_iterations = atoi(argv[1]);
      if ( (max_iterations <= 0) || (max_iterations > 20) )
      max_iterations = 10;
   }
   pid = fork();
   
   // Unable to create child process
   if (pid < 0) {
      perror("fork error\n");
      exit(1);
   }
   
   // Child process
   if (pid == 0) {
      fd = open("/tmp/DAEMON.txt", O_WRONLY|O_CREAT|O_TRUNC, 0644);
      if (fd == -1) {
         perror("daemon txt file open error\n");
         return 1;
      }
      printf("Child: pid is %d and ppid is %d\n", getpid(), getppid());
      printf("\nChild process before becoming session leader\n");
      sprintf(buffer, "ps -ef|grep %s", argv[0]);
      system(buffer);
      setsid();
      printf("\nChild process after becoming session leader\n");
      sprintf(buffer, "ps -ef|grep %s", argv[0]);
      system(buffer);
      close(STDIN_FILENO);
      close(STDOUT_FILENO);
      close(STDERR_FILENO);
   } else {
      printf("Parent: pid is %d and ppid is %d\n", getpid(), getppid());
      printf("Parent: Exiting\n");
      exit(0);
   }
   
   // Executing max_iteration times
   for (counter = 0; counter < max_iterations; counter++) {
      sprintf(buffer, "Daemon process: pid is %d and ppid is %d\n", getpid(), getppid());
      write(fd, buffer, strlen(buffer));
      sleep(2);
   }
   strcpy(buffer, "Done\n");
   write(fd, buffer, strlen(buffer));
   
   // Can't print this as file descriptors are already closed
   printf("DoneDone\n");
   close(fd);
   return 0;
}
Parent: pid is 193524 and ppid is 193523
Parent: Exiting
4581875  193525      0  0 09:23  ?      00:00:00 main
4581875  193526 193525  0 09:23  ?      00:00:00 sh -c ps -ef|grep main
4581875  193528 193526  0 09:23  ?      00:00:00 grep main
4581875  193525      0  0 09:23  ?      00:00:00 main
4581875  193529 193525  0 09:23  ?      00:00:00 sh -c ps -ef|grep main
4581875  193531 193529  0 09:23  ?      00:00:00 grep main