Linux中的特殊进程(孤儿进程、僵尸进程、守护进程)

一、孤儿进程

1)父进程退出,子进程不退出,此时子进程被1号(init)进程收养,变成孤儿进程。

2)孤儿进程会脱离终端控制,且运行在后端,不能用ctrl+c杀死后端进程,但是可以被kill -9杀死。

测试:

cs 复制代码
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main(int argc, const char *argv[])
{
    //父进程退出 子进程不退出
    pid_t cpid = fork();
    if(cpid > 0)
    {
    }
    else if(0 == cpid)
    {
        while(1)                                                      
        {
            printf("this is child %d %d\n", getppid(), getpid());
            sleep(1);
        }
    }
    else
    {
        perror("fork");
        return -1;
    }

    return 0;
}

二、僵尸进程

子进程退出,父进程不退出去,且父进程没有给子进程收尸,此时子进程就变成僵尸进程。

注意:

1)僵尸进程只能被回收,不能被杀死。

2)僵尸进程有危害:占用进程号,占用部分内存空间,占用物理空间,占用进程调度块(PCB)等等...

3)回收僵尸进程的方式:

  • 退出父进程后,子进程的资源由内核自动回收。
  • wait / waitpid函数回收。缺点:阻塞函数,父进程无法做自己的事情。非阻塞形式,有可能收不到。
  • 结合信号的方式回收僵尸进程:当子进程退出后,通知父进程收尸。

测试:

cs 复制代码
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main(int argc, const char *argv[])
{
    //子进程退出,父进程不退出
    pid_t cpid = fork();
    if(0 == cpid)
    {
    }
    else if(cpid > 0)                                        
    {
        while(1)
        {
            printf("this is parent %d %d\n", getpid(), cpid);
            sleep(1);
        }
    }
    else
    {
        perror("fork");
        return -1;
    }

    return 0;
}

三、守护进程【幽灵进程】

  1. 守护进程脱离于终端,且运行在后端。
  2. 守护进程在执行过程中不会将信息显示在任何终端上,避免影响前端任务执行。且不会被任何终端产生的终端信息所打断。
  3. 守护进程目的:需要周期性执行某个任务或者周期性等待处理某些事情的时候,为了避免影响前端执行或者被前端信息打断的时候,可以使用守护进程。

守护进程的创建(5步):

  1. 创建孤儿进程:所有工作都在子进程中执行,从形式上脱离终端控制。

    cs 复制代码
    fork(), 退出父进程
  2. 创建新的会话组:使子进程完全独立出来,防止兄弟进程对其有影响。

    cs 复制代码
    setsid() 函数
    功能:创建一个新的进程组和会话组,成为该进程组和会话组组长;
    原型:
           #include <sys/types.h>
           #include <unistd.h>
    
           pid_t setsid(void);
  3. 修改当前孤儿进程的运行目录为不可卸载的文件系统:例如根目录,/tmp;防止运行目录被删除后,导致进程崩溃

    cs 复制代码
    chdir函数
    功能:修改运行目录;
           #include <unistd.h>
    
           int chdir(const char *path);
    chdir("/");
    
    注意:从当前位置往后,运行在指定的目录下;
  4. 重设文件权限掩码:umask(0), 一般清零;

  5. 关闭所有文件描述符,从父进程继承过来的文件描述符不会用到,浪费资源。

代码:

cs 复制代码
 #include <stdio.h>
 #include <stdlib.h>
 #include <sys/types.h>
 #include <sys/wait.h>
 #include <sys/stat.h>
 #include <unistd.h>
 int main(int argc, const char *argv[])
 {
     //创建孤儿进程
     pid_t cpid = fork();
     if(0 == cpid)
     {
         //创建新的回话
         pid_t sid =setsid();
         printf("sid = %d\n",sid);
         //修改运行目录为不可卸载的文件目录下
         chdir("/");
         //清空文件权限掩码
         umask(0);
         //关闭所有文件描述符
         for(int i = 0;i<getdtablesize();i++)
             close(i);
 
         while(1)
         {
             //守护进程运行的周期性代码
             sleep(1);
         }
     }
     return 0;
 }
  • 输入ps ajx可查看进程之间的关系;
  • 输入pidof 文件名 可查看该文件是否有进程在运行;
  • 可通过 【kill -9 进程的pid号】将这个进程杀死;
  • 也可通过【killall -9 文件名】将该文件的所有在运行的进程杀死。
相关推荐
milo.qu4 小时前
RockyLinux9.7 docker部署Jisti Meet
linux·docker·容器
GanGanGanGan_4 小时前
CentOS 7.9 glibc 2.17 源码编译升级到 glibc 2.31
linux·运维·centos·glibc
charlie1145141914 小时前
嵌入式Linux驱动开发——class 和 device 模型 - 自动创建设备节点的幕后机制
linux·运维·驱动开发
枳实-叶6 小时前
【Linux驱动开发】第四天:dmesg日志全解+驱动加载失败极速排查
linux·运维·驱动开发
wdfk_prog7 小时前
正常关闭虚拟机时,不要点“关机”,而要点“关闭客户机”
linux·c语言·网络·ide·vscode
爱吃土豆的马铃薯ㅤㅤㅤㅤㅤㅤㅤㅤㅤ7 小时前
通过java后端代码来实现给word内容补充格式文本内容控件,以及 设置控件的标记和标题
java·c#·word
fish_xk8 小时前
Linux开方工具
linux·运维·服务器
流年如夢8 小时前
单链表 -->增、删、查、改等详细操作
c语言·数据结构
用户23678298016810 小时前
Linux find 命令深度解析:从递归遍历到性能优化的完整实现
linux
ascarl201010 小时前
Linux.do 帖子整理:AI 调用 Chrome DevTools 调试前端页面
linux·前端·人工智能