如何写一个linux精灵进程

本文详细介绍了Linux守护进程(精灵进程)的定义及其重要性,并逐步解析了创建守护进程的步骤,包括umask设置、两次fork、改变工作目录、关闭不必要的文件描述符以及忽略SIGCHLD信号等。通过实例代码展示了如何实现一个基本的守护进程,帮助读者深入理解这一关键的系统服务执行方式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

  • 什么是精灵进程
    精灵进程也称守护进程(Daemon):是运行在后台的一种特殊进程,它独立于控制终端并周期性的执行某种任务,或等待处理某些发生的事件。Linux大多数服务器就是用精灵进程实现的。
    1.我们先来观察一些系统中的精灵进程
    这里写图片描述
    精灵进程不具有控制终端所以TTY-?,终端前台进程组的id为-1。

  • 自己写一个精灵进程
    根据精灵进程的概念,我们可以总结出实现一个精灵进程的基本步骤。

1.调用umask将文件模式创建屏蔽字设为0
2.调用fork()父进程退出
3.调用setsid创建一个新会话
4.再次fork()父进程退出
5.将当前工作目录更改为根目录
6.关闭不需要的文件描述符
7.忽略SIGCHLD信号

接下来让我们一步一步来实现一个精灵进程。
1.调用umask将文件模式创建屏蔽字设为0

umask(0);

由继承得来的文件方式创建屏蔽字可能会拒绝设置某些许可权。在这种情况下如果精灵进程想要创建一个可读可写的文件,而文件方式创建屏蔽字屏蔽了读写的许可权,那么精灵进程创建的文件就无法获得读写权限。
2.调用fork()父进程退出

pid_t id = fork();
    if(id < 0)
        exit(EXIT_FAILURE);
    if(id > 0)
        exit(EXIT_SUCCESS);

精灵进程没有控制终端,为了不使其成为孤儿进程需要调用setsid()函数获得一个SID。
执行setsid()函数的结果

1.创建一个新的会话,当前进程成为会话首进程,SID就是该进程的pid。
2.创建一个新的进程组,当前进程成为组长进程PGID就是该进程的pid。
3.如果当前进程原本有一个控制终端,则它失去这个控制终端。
4.调用该函数成功返回一个新的会话id失败返回-1,调用该函数之前该进程不能是该进程组的组长进程。

为了满足性质4我们可以调用fork创建一个子进程,子进程不是进程组的第一个进程所以一定不会是组长进程,我们用子进程调用setsid()函数。
3.调用setsid创建一个新会话

pid_t sid = setsid();
    if(sid < 0)
        exit(EXIT_FAILURE);

4.再次fork()父进程退出

id = fork();
    if(id < 0)
        exit(EXIT_FAILURE);
    if(id != 0)
        exit(EXIT_SUCCESS);

会话首进程仍然可以打开一个终端,为了保证后续不不会再打开终端,再次fork()保证精灵进程不再是会话首进程。

5.将当前工作目录更改为根目录

if(chdir("/") < 0)
        exit(EXIT_FAILURE);

从父进程继承过来的工作目录,可能在一个装配的文件系统中,因为精灵进程通常在系统中是一直存在的,如果当前工作目录在一个装配的文件系统中,那么该文件系统不能被拆卸。

6.关闭不需要的文件描述符

 close(0);
 close(1);
 close(2);

这样精灵进程不在持有从父进程继承来的文件描述符。

7.忽略SIGCHLD信号

signal(SIGCHLD,SIG_IGN);

这样fork出来的子进程在执行完毕后自动清理不会产生僵尸进程。

  • 完整代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<signal.h>
void Daemon()
{
    umask(0);
    pid_t id = fork();
    if(id < 0)
        exit(EXIT_FAILURE);
    if(id > 0)
        exit(EXIT_SUCCESS);
    pid_t sid = setsid();
    if(sid < 0)
        exit(EXIT_FAILURE);
    signal(SIGCHLD,SIG_IGN);
    id = fork();
    if(id < 0)
        exit(EXIT_FAILURE);
    if(id != 0)
        exit(EXIT_SUCCESS);
    if(chdir("/") < 0)
        exit(EXIT_FAILURE);
    close(0);
    close(1);
    close(2);
}
int main()
{
    Daemon();
    while(1);
    return 0;

}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值