什么是守护进程?

什么是守护进程?在了解守护进程之前,需要先知道什么是什么是终端?什么是作业?什么是进程组?什么是会话?在Linux中,每一个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程都会依附于这个终端,这个

大家好,又见面了,我是你们的朋友全栈君。

在了解守护进程之前,需要先知道什么是什么是终端?什么是作业?什么是进程组?什么是会话?

在 Linux 中,每一个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程都会依附于这个终端,这个终端就称为这些进程的控制终端,当控制终端被关闭时,相应的进程都会自动关闭。

守护进程就是一个生存周期较长,独立于控制终端并且周期性执行某种任务的进程。之所以要脱离终端,就是为了防止进程运行过程中被任何终端信息所打断

所以,要创建守护进程,我们就要将这个进程脱离终端。

shell分前后台作业来控制的不是进程而是作业。一个作业由多个进程组成。Shell可以运行一个前台作业和任意多个后台作业,称为作业控制。bash就是一个独立的作业。

进程组是一个或多个进程的集合,每个进程除了有一个PID以外,还有一个PGID。PGID就是组长的PID。进程组通常和一个作业相关联,可以接收来自同一个终端的信号。

当然,进程组和作业也并不是完全等价的两个概念:如果作业中某个进程有创建了新的子进程,该子进程不属于作业,但属于该进程组。

会话(Session)是一个或多个进程组的集合。一个会话可以有一个控制终端。一个会话中,有一个前台作业和若干个后台作业。会话SID是会话手进程的PID。

为什么只能运行一个前台作业?当我们在前台新起了一个作业,shell就被提到了后台,因此shell就没有办法再继续接受我们的指令并且解析运行了。但是如果前台进程退出了,shell就会有被提到前台来,就可以继续接受我们的命令并且解析运行。

那么,如何来切断进程和终端的关系呢?

首先,调用 setsid() 使子进程成为新的会话组长。setsid() 调用成功后,进程成为新的会话组长和新的进程组长,并与原来的登录会话和进程组脱离。

调用setsid()有一个前提,就是该进程不能是一个组长进程,因此需要先fork并且杀死父进程,setsid ()的调用者是子进程。

接下来,要禁止进程重新打开控制终端。能打开控制终端的进程一定是进程组组长,因此我们需要再次fork(),并且杀死父进程,留下的子进程就不再是话首进程和进程组组长。于是,这个子进程也不再拥有打开终端的权限,至此,我们彻底切断了该进程和终端的联系。

最后,要关闭打开的文件描述符,或者对打开的文件描述符进行重定向。因为进程会继承从父进程那里的文件描述符,如果不关闭,会浪费系统的资源。

如果想改变该进程的所在目录,可以调用chdir(“/”) 将该守护进程转移到根目录。

如果该守护进程有子进程,那么守护进程需要等待子进程退出,否则子进程会变成僵尸进程。为了减少该守护进程的负担,防止其回收子进程对服务器并发性能的影响,可以使用signal(SIGCHLD, SIG_IGN) 对SIGCHLD忽略。这样就可以防止僵尸进程产生。

#include <unistd.h>   
#include <signal.h>   
#include <fcntl.h>  
#include <sys/syslog.h>  
#include <sys/param.h>   
#include <sys/types.h>   
#include <sys/stat.h>   
#include <stdio.h>  
#include <stdlib.h>  
#include <time.h>  
  
int init_daemon(void)  
{   
    int pid;   
    int i;  
      
    // 1)屏蔽一些控制终端操作的信号  
    signal(SIGTTOU,SIG_IGN);   
    signal(SIGTTIN,SIG_IGN);   
    signal(SIGTSTP,SIG_IGN);   
    signal(SIGHUP ,SIG_IGN);  
   
    // 2)在后台运行  
    if( pid=fork() ){ // 父进程  
        exit(0); //结束父进程,子进程继续  
    }else if(pid< 0){ // 出错  
        perror("fork");  
        exit(EXIT_FAILURE);  
    }  
      
    // 3)脱离控制终端、登录会话和进程组  
    setsid();    
      
    // 4)禁止进程重新打开控制终端,这是一种防御性编程,是可选的一步
    if( pid=fork() ){ // 父进程  
        exit(0);      // 结束第一子进程,第二子进程继续(第二子进程不再是会话组长)
     
    }else if(pid< 0){ // 出错  
        perror("fork");  
        exit(EXIT_FAILURE);  
    }    
      
    // 5)关闭打开的文件描述符  
    // NOFILE 为 <sys/param.h> 的宏定义  
    // NOFILE 为文件描述符最大个数,不同系统有不同限制  
    for(i=0; i< NOFILE; ++i){  
        close(i);  
    }  
      
    // 6)改变当前工作目录  
    chdir("/tmp");   
      
    // 7)重设文件创建掩模,因为进程从创建它的父进程那里继承了文件创建掩模。它可能修改守护进程所创建的文件的存取权限。
    umask(0);    
      
    // 8)处理 SIGCHLD 信号  
    signal(SIGCHLD,SIG_IGN);  
      
    return 0;   
}   
  
int main(int argc, char *argv[])   
{  
    init_daemon();  
      
    while(1);  
  
    return 0;  
} 
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/154705.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • java类加载过程详解_java三个类加载器

    java类加载过程详解_java三个类加载器朋友给我发了一道有意思的题目,如下为什么用.class的方式加载类和以Class.forName()的方式加载的结果不同呢,原因很简单,就是类加载过程的不同。这就扯到基础理解上了,就是Java是如何加载一个类的呢?上图是我绘制的整个Java类加载过程。首先是编译期,将Java源文件也就是敲好的代码通过编译,转换成.class文件,也就是字节码文件(byte),然后经过传输传…

    2022年8月11日
    4
  • SpringBoot +JDBC连接Mysql数据库

    SpringBoot +JDBC连接Mysql数据库SpringBoot使用JDBC连接Mysql数据库    Spring连接Mysql的方式有很多,例如JDBC,SpringJPA,Hibeirnate,Mybatis等,本文主要介绍使用最简单、最底层的JDBC方式来连接Mysql数据库,JDBC连接数据库,主要是注入JdbcTemplate,使用JdbcTemplate来操作数据库。一、在mysql中的test库中建立user表,…

    2022年6月25日
    35
  • 软件测试面试问题及答案_中软国际测试面试笔试题

    软件测试面试问题及答案_中软国际测试面试笔试题文章目录软测面试题附答案<二>二、接口面试题1、你们公司的接口测试流程是怎样的?2、接口测试里面如何分析一个bug是前端还是后端的?3、Jmeter参数化的方式有哪几种可以实现4、在接口测试中关联是什么含义?一般什么场景需要用到,postman中是怎样用的?5、常用的性能指标名称与具体含义?6、没有接口和文档怎么进行工作?三、自动化面试题1、Selenium中有哪几种元素定位方式?2、隐式等待与显式等待有什么不同?3、自动化测试里面遇到验证码,一般怎么处理比较好?四、安全1、什么是SQL注入攻击

    2022年9月29日
    0
  • navicat mac激活码-激活码分享

    (navicat mac激活码)这是一篇idea技术相关文章,由全栈君为大家提供,主要知识点是关于2021JetBrains全家桶永久激活码的内容https://javaforall.net/100143.htmlIntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,上面是详细链接哦~23LNPMIJZT-eyJsaWNlb…

    2022年3月29日
    92
  • javascript 类数组概念详解

    javascript 类数组概念详解

    2021年6月7日
    99
  • c++ 线程间通信方式「建议收藏」

    c++ 线程间通信方式「建议收藏」线程同步和线程互斥互斥:是指某一资源同时只允许一个访问者对其进行访问,具有唯一性和排它性。但互斥无法限制访问者对资源的访问顺序,即访问是无序的,线程间不需要知道彼此的存在。同步:是指在互斥的基础上(大多数情况),通过其它机制实现访问者对资源的有序访问,线程间知道彼此的存在。在大多数情况下,同步已经实现了互斥,特别是所有写入资源的情况必定是互斥的。少数情况是指可以允许多个访问者同时访问资源线程…

    2022年10月7日
    1

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号