进程间关系和守护进程

1.进程组

    在Linux下,每一个进程都有自己的一个进程ID,但是他还包含了一个进程组ID,它是属于一个进程组的。那么进程组是什么呢? 

    进程组是由一个或多个的进程组成的集合。每个进程组有一个唯一的进程组ID,也可以说是这个进程组的编号。那么既然是一个组,就要有一个组长进程,既然是组长进程,那么该进程组的组ID就等于组长进程的ID。

    组长进程可以创建一个进程组,并且创建该组中的进程,同时也拥有权力去终止创建的进程。

    在一个进程组中,如果组长进程不存在,但是该组中还有别的进程存在,那么该进程组就一直存在,与组长进程是否存在没有关系。

创建了三个进程


可以看到他们的PGID是一样的,是第一个进程(也就是组长进程)的PID

杀死组长进程后,该组仍旧存在


2.作业

    Shell分前后台来进行控制的不是进程而是作业或者是进程组。

    一个前台作业和一个后台作业都可以由多个进程组成。

    作业控制 : 在Shell里,可以运行一个前台作业和多个后台作业

作业与进程组的区别:如果作业中的某个进程又创建了子进程,子进程不属于作业,但是属于进程组

    如果某个作业一旦结束,Shell就会把自己提到前台,但是原来的进程组创建的子进程还在,可是子进程并不属于作业,这个时候    如果原来的前台进程没有运行完毕(其实这里说的是子进程)的话,会自动变为后台进程组(前提是该子进程得存在没有被终止)

        或者可以这样说,当前台有某个作业运行的时候,因为前台只能运行一个作业,这个时候Shell就会自动被运行到后台。当这个前台作业运行完毕后,Shell就会被提到前台,可以继续接受用户的输入


创建一个进程然后fork出一个子进程,i为0的时候,父进程会退出,但是子进程还在运行,但是此时的前台作业因为父进程的退出变成了Shell,子进程被自动提到了后台,此时可以显示Shell了

但是父进程也就是组长进程虽然退出了,但是子进程所属的组还在,然后杀掉即可




3.会话

    会话的概念 : 会话是一个或者多个进程组的集合。

    一个会话可以有一个控制终端。

    控制进程:建立与控制终端连接的会话首进程

    会话包含的内容 : 会话首进程(控制进程)、一个前台进程组和一个或多个后台进程组


在这里我们创建了三个进程,它们属于同一个进程组,而且它们的SID是一样的,SID是会话id,所以属于同一个会话,而这个会话ID1发现是bash(解释器),也就是说bash是会话首进程,并且是这三个进程的的父进程。

4.作业控制

    在Shell里面,分前后台进行控制的不是进程而是作业或者是进程组

    一个前台作业可以由多个进程组组成,一个后台作业也可以由多个进程组组成

    作业控制:在Shell里面,可以同时运行一个前台作业和任意多个后台作业便称之为作业控制

新建两个作业至后台


jobs可以查看当前有哪些作业


fg将作业1提至前台运行,然后ctrl+z可以将其提至后台,将其提至后台那么它在前台的运行就停止了,这个时候jobs可以看到前台作业1已经stop了


bg命令可以使某个停止的作业继续在后台运行,那么在前台终止的作业1提至后台后,用bg就又处于运行状态了

注意:前台作业不能被stop

5.守护进程

    守护进程(精灵进程),它是一种运行在后台的特殊进程,它独立于终端并且7*24小时的运行

    守护进程是一种很有用的进程。在Linux上大多数的服务器就是用守护进程来实现的,例如有:ssh服务器、Web服务器HTTP,FTP服务器等等,并且守护进程还能完成许多系统任务,如可以规划进程crond。

    在Linux下,当我们进行启动时,同时也会有许多的系统服务进程也进行启动,不能直接和用户进行交互。

    别的进程都是在用户登录或者程序在运行的时候进行创建,在程序运行结束的时候或者用户注销时终止。

    守护进程并不受用户登录或者注销的影响,它们一直在运行着

    >>>守护进程的标志 : 凡是在TPGID一栏写着-1的都是没有控制终端的进程(守护进程)

                                 守护进程通常采用以d结尾的名字,表示Daemon

                                      在COMMEND一列用【】括起来的名字表示内核线程,通常采                                           用 Kernel表示

   >>> 创建守护进程 : 最关键的一步是调用setid函数创建一个新的Session,并成为Session Leader。

        接口 :

#include <unsitd.h>

pid_t stdsid(void);  //函数调用成功的时候返回新创建的进程id,出错返回-1

    注意 : 在调用这个函数之前,当前进程是不允许是进程组的组长,否则函数会返回-1

                要保证当前进程不是进程组的组长,可以调用fork在调用sedsid。    

  1.#include <stdio.h>
  2 #include <signal.h>
  3 #include <unistd.h>
  4 #include <stdlib.h>
  5 #include <sys/stat.h>
  6 #include <fcntl.h>
  7 
  8 void mydaemon( void )
  9 {
 10     int i = 0;
 11     pid_t pid;
 12     int fd0;
 13     struct sigaction sa;
 14 
 15     umask(0); // 调用umask将文件模式创建屏蔽子设置为0
 16 
 17     //2.调用fork,使得父进程推出
 18 
 19     //如果守护进程是一条简单的shell命令启动的,当父进程终止后,那么shell就认为
 20     //该命令已经执行完毕
 21 
 22     //保证子进程不是一个进程组的组长进程
 23     if( (pid = fork()) < 0 ){
 perror("fork()");
 25     }
 26     else if( pid > 0 ){
 27         exit(0); // 父进程终止
 28     }
 29 
 30     sa.sa_handler = SIG_IGN; // 忽略SIGCHLD信号
 31     setsid();//创建一个新的会话(子进程)
 32     if(sigaction(SIGCHID,&sa,NULL) < 0)
 33         return;
 34 
 35 
 36     close(0);
 37     fd0 = open("/dev/null",O_RDWR);
 38     dup2(fd0,1);
 39     dup2(fd0,2);
结果:


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值