进程组、会话、守护进程

(一).进程组

进程组,也称为作业,代表一个或多个进程的集合。并且每个进程都属于一个进程组。当我们调用fork()函数创建子进程时,默认子进程与父进程属于同一个进程组,并且进程组ID等于组长进程ID,默认第一个进程是组长ID。进程组组长可以创建一个进程组以及创建该组的进程,只要该进程组中还有进程存在,那这个进程组就存在,不管组长进程有没有终止。

相关函数:

1.获取当前进程的组ID:
函数原型:pid_t getpgrp(void)
返回值:返回当前进程的进程组ID
2.获取指定进程的进程组ID:
函数原型:pid_t getpgid(pit_t pid)
返回值:成功返回指定进程的进程组ID,失败返回-1,设置errno
参数:pid:要查看的进程的进程号
3.改变进程的进程组:
函数原型:int setpgid(pid_t pid, pid_t pgid)
返回值:成功返回0,失败返回-1,并设置errno
参数:pid:要改变的进程号;pgid:要设置的进程组ID。意思是,将进程号为pid的进程的进程组ID设置为pgid,如果pgid对应的进程组不存在,那么就会创建一个进程组。

(二).会话

会话(session)是一个或多个进程组的集合

相关函数:

1.创建会话:
函数原型:pid_t setsid(void)
返回值:返回创建的会话ID

创建一个会话需要注意以下几点:
1).调用创建会话的进程不能是进程组组长,如果是,会报错返回。
2).该进程会变成新会话的首进程(意思就是是当前会话唯一一个进程)
3).新会话会丢弃原来的控制终端,处于没有控制终端状态
4).建立新会话时,一般的做法是,先调用fork,然后让父进程终止,子进程则继续。这是因为,进程组ID肯定是等于父进程ID的,而子进程的ID是重新分配的,这样就保证了子进程不是一个进程组的组长。

2.获取进程所属的会话ID:
函数原型:pid_t getsid(pid_t pid)
返回值:返回进程所属的会话ID
参数:pid:要查看的进程ID。当pid=0时,表示查看当前进程的会话ID。ps ajx命令可以查看系统中的进程。参数a表示不仅列出当前用户的进程,也列出所有其它用户的进程,参数x表示不仅列有控制终端的进程,也列出所有无控制终端的进程,参数j表示列出与作业控制相关的信息。

(三).守护进程

守护进程(Daemon)时生存期长的一种进程,并且在后台运行(无控制终端),一般会在系统引导的时候启动,在系统关闭的时候终止。一般采用以d结尾的名字。
要想完成一个守护进程,需要有这样6个步骤:
1.创建子进程,父进程退出
2.子进程创建新的会话
3.改变当前目录为根目录:用chdir()函数实现。这是因为子进程从父进程继承过来的工作目录可能在一个挂在的文件系统中,而守护进程工作的目录不允许卸载,这就导致了占用了不必要的系统资源。
4.重新设置文件权限掩码:用umask()函数实现。为了防止创建的文件拒绝了读/写权限,造成守护进程无法访问。
5.关闭文件描述符:子进程会继承父进程的文件描述符表,而守护进程用不到这些,所以关了节约资源。
6.开始执行守护进程的工作

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
void my_daemond()
{
    pid_t pid;
    pid = fork();
    if(pid < 0)
    {
        perror("fork error");
        exit(1);
    }
    else if(pid > 0)        //关闭父进程
        exit(2);
    setsid();               //由子进程创建新的会话
    if(chdir("/") < 0)      //改变工作目录
    {
        perror("chdir");
        exit(1);
    }
    umask(0);               //更改umask掩码
    close(STDIN_FILENO);    //关闭0号文件描述符
    int fd = open("/dev/null", O_RDWR); //打开"/dev/null“,占用0号文件描述符
    dup2(fd, STDOUT_FILENO);        //将标准输出重定向到/dev/null
    dup2(fd, STDERR_FILENO);        //将标准错误重定向到/dev/null

}
int main()
{
    my_daemond();
    while(1);
}

编译运行之后,使用ps ajx命令会看到该进程。查看ppid那一列,会发现该进程的父进程变成了1,这说明它成了一个孤儿进程。孤儿进程是指,父进程fork出了子进程,当父进程终止时,这个子进程就成了孤儿进程,它的父进程会成为init进程(pid为1)。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值