如何写一个linux精灵进程

来源:互联网 发布:ecshop小京东数据字典 编辑:程序博客网 时间:2024/05/20 23:59
  • 什么是精灵进程
    精灵进程也称守护进程(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;}
  • 执行结果
    这里写图片描述

  • 完整代码请戳:
    https://coding.net/u/Hyacinth_Dy/p/MyCode/git/blob/master/%E7%B2%BE%E7%81%B5%E8%BF%9B%E7%A8%8B

0 0
原创粉丝点击