中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

詳解linux下避免僵尸進程的幾種方法

發布時間:2020-09-25 07:53:43 來源:腳本之家 閱讀:170 作者:ComingFlying 欄目:服務器

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;
}

方法三:

#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");
  }
}

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

祁门县| 阿城市| 什邡市| 抚州市| 神农架林区| 芒康县| 西华县| 铜山县| 嘉黎县| 绥棱县| 罗平县| 洛宁县| 华亭县| 蒙阴县| 曲松县| 偏关县| 大英县| 康马县| 高邑县| 繁昌县| 栖霞市| 沅江市| 新田县| 从江县| 和平县| 鄄城县| 巴彦县| 石林| 图木舒克市| 奈曼旗| 庄河市| 屯门区| 华亭县| 潜山县| 临泽县| 罗山县| 应用必备| 白沙| 上蔡县| 桓仁| 泗水县|