deamon(守护进程)的实现

文章来源:http://blog.csdn.net/xuleilx/article/details/8258798

感谢博主

Daemon

  Daemon程序是一直运行的服务端程序,又称为守护进程。通常在系统后台运行,没有控制终端不与前台交互,Daemon程序一般作为系统服务使用。Daemon是长时间运行的进程,通常在系统启动后就运行,在系统关闭时才结束。一般说Daemon程序在后台运行,是因为它没有控制终端,无法和前台的用户交互。Daemon程序一般都作为服务程序使用,等待客户端程序与它通信。我们也把运行的Daemon程序称作守护进程

Daemon程序实现方法

下面是编写守护进程的基本过程:
1)屏蔽一些控制终端操作的信号

这是为了防止守护进行在没有运行起来前,控制终端受到干扰退出或挂起。关于信号的更详细用法。
    signal(SIGTTOU,SIG_IGN);   
    signal(SIGTTIN,SIG_IGN);   
    signal(SIGTSTP,SIG_IGN);   
    signal(SIGHUP ,SIG_IGN);  


2)在后台运行

这是为避免挂起控制终端将守护进程放入后台执行。方法是在进程中调用 fork() 使父进程终止, 让守护进行在子进程中后台执行。

    if( pid = fork() ){ // 父进程  
        exit(0);        //结束父进程,子进程继续  
    }  

3) 脱离控制终端、登录会话和进程组

有必要先介绍一下 Linux 中的进程与控制终端,登录会话和进程组之间的关系:进程属于一个进程组,进程组号(GID)就是进程组长的进程号(PID)。登录会话可以包含多个进程组。这些进程组共享一个控制终端。这个控制终端通常是创建进程的 shell 登录终端。 控制终端、登录会话和进程组通常是从父进程继承下来的。我们的目的就是要摆脱它们 ,使之不受它们的影响。因此需要调用 setsid() 使子进程成为新的会话组长,示例代码如下:

setsid();  

setsid() 调用成功后,进程成为新的会话组长和新的进程组长,并与原来的登录会话和进程组脱离。由于会话过程对控制终端的独占性,进程同时与控制终端脱离。 
4)禁止进程重新打开控制终端

现在,进程已经成为无终端的会话组长,但它可以重新申请打开一个控制终端。可以通过使进程不再成为会话组长来禁止进程重新打开控制终端,采用的方法是再次创建一个子进程,示例代码如下:
    if( pid=fork() ){ // 父进程  
        exit(0);      // 结束第一子进程,第二子进程继续(第二子进程不再是会话组长)   
    }  

5)关闭打开的文件描述符

进程从创建它的父进程那里继承了打开的文件描述符。如不关闭,将会浪费系统资源,造成进程所在的文件系统无法卸下以及引起无法预料的错误。按如下方法关闭它们:

    // NOFILE 为 <sys/param.h> 的宏定义  
    // NOFILE 为文件描述符最大个数,不同系统有不同限制  
    for(i=0; i< NOFILE; ++i){// 关闭打开的文件描述符  
        close(i);  
    }  

6)改变当前工作目录


进程活动时,其工作目录所在的文件系统不能卸下。一般需要将工作目录改变到根目录。对于需要转储核心,写运行日志的进程将工作目录改变到特定目录如 /tmp。示例代码如下:

    chdir("/");  

7)重设文件创建掩模

进程从创建它的父进程那里继承了文件创建掩模。它可能修改守护进程所创建的文件的存取权限。为防止这一点,将文件创建掩模清除:
    umask(0);  

8)处理 SIGCHLD 信号

但对于某些进程,特别是服务器进程往往在请求到来时生成子进程处理请求。如果父进程不等待子进程结束,子进程将成为僵尸进程(zombie)从而占用系统资源。如果父进程等待子进程结束,将增加父进程的负担,影响服务器进程的并发性能。在 Linux 下可以简单地将 SIGCHLD 信号的操作设为 SIG_IGN 。关于信号的更详细用法。
signal(SIGCHLD, SIG_IGN);  
   另一个问题是Daemon程序不能和终端交互,也就无法使用printf方法输出信息了。我们可以使用syslog机制来实现信息的输出,方便程序的调试。在使用syslog前需要首先启动syslogd程序,关于syslogd程序的使用请参考它的man page,或相关文档,我们就不在这里讨论了。

Daemon程序实现例子

  一个Daemon程序的例子 编译运行环境为Redhat Linux 8.0。  我们新建一个daemontest.c程序,文件内容如下:
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <stdlib.h>
#include <stdio.h>
#include <syslog.h>
#include <signal.h>

int daemon_init(void)
{ 
    pid_t 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);
    }
    printf("%d\n", NOFILE);
    
    // 6)改变当前工作目录
    chdir("/tmp"); 
    
    // 7)重设文件创建掩模
    umask(0);  
    
    // 8)处理 SIGCHLD 信号
    signal(SIGCHLD,SIG_IGN);
    
    return 0; 
}  

void sig_term(int signo)  
{ 
    if(signo == SIGTERM)  
    /* catched signal sent by kill(1) command */  	
    {
        syslog(LOG_INFO, "program terminated.");  	
        closelog(); 
        exit(0); 
    }  
}  
int main(void)  
{ 
    if(daemon_init() == -1)  
    { 
        printf("can't fork self\n"); exit(0); 
    }  
    openlog("daemontest", LOG_PID, LOG_USER);  	
    syslog(LOG_INFO, "program started.");  
    signal(SIGTERM, sig_term); /* arrange to catch the signal */  
    while(1) 
    { 
        sleep(1); /* put your main program here */ 
    }  
    return(0); 
}

 使用如下命令编译该程序: gcc -Wall -o daemontest daemontest.c编译完成后生成名为daemontest的程序,执行./daemontest来 测试程序的运行。  使用ps axj命令可以显示系统中已运行的daemon程序的信息,包括进程ID、session ID、控制终端等内容。  部分显示内容:   PPID PID PGID SID TTY TPGID STAT UID TIME COMMAND  1098 1101 1101 1074 pts/1 1101 S 0 0:00 -bash 1 1581 777 777 ? -1 S 500 0:13 gedit 1 1650 1650 1650 ? -1 S 500 0:00 ./daemontest 794 1654 1654 794 pts/0 1654 R 500 0:00  ps axj 从中可以看到daemontest程序运行的进程号为1650。  我们再来看看/var/log/messages文件中的信息: Apr 7 22:00:32 localhost  daemontest[1650]: program started.  我们再使用kill 1650命令来杀死这个进程,/var/log/messages文件中就会有如下的信息:  Apr 7 22:11:10 localhost daemontest[1650]: program terminated.

  使用ps axj命令检查,发现系统中daemontest进程已经没有了。


--------------------------------------------------------------------------------------------------------------


守护进程

百科名片

在linux或者unix操作系统中在系统的引导的时候会开启很多服务,这些服务就叫做守护进程。为了增加灵活性,root可以选择系统开启的模式,这些模式叫做运行级别,每一种运行级别以一定的方式配置系统。 守护进程是脱离于终端并且在后台运行的进程。守护进程脱离于终端是为了避免进程在执行过程中的信息在任何终端上显示并且进程也不会被任何终端所产生的终端信息所打断。

守护进程简介

  守护进程,也就是通常说的Daemon进程,是Linux中的后台服务进程。它是一个生存期较长的进程,通常独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。守护进程常常在系统引导装入时启动,在系统关闭时终止。Linux系统有很多守护进程,大多数服务都是通过守护进程实现的,同时,守护进程还能完成许多系统任务,例如,作业规划进程crond、打印进程lqd等(这里的结尾字母d就是Daemon的意思)。  由于在Linux中,每一个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程都会依附于这个终端,这个终端就称为这些进程的控制终端,当控制终端被关闭时,相应的进程都会自动关闭。但是守护进程却能够突破这种限制,它从被执行开始运转,直到整个系统关闭时才退出。如果想让某个进程不因为用户或终端或其他地变化而受到影响,那么就必须把这个进程变成一个守护进程。

创建守护进程

创建子进程,父进程退出

  这是编写守护进程的第一步。由于守护进程是脱离控制终端的,因此,完成第一步后就会在Shell终端里造成一程序已经运行完毕的假象。之后的所有工作都在子进程中完成,而用户在Shell终端里则可以执行其他命令,从而在形式上做到了与控制终端的脱离。  在Linux中父进程先于子进程退出会造成子进程成为孤儿进程,而每当系统发现一个孤儿进程时,就会自动由1号进程(init)收养它,这样,原先的子进程就会变成init进程的子进程。

在子进程中创建新会话

  这个步骤是创建守护进程中最重要的一步,虽然它的实现非常简单,但它的意义却非常重大。在这里使用的是系统函数setsid,在具体介绍setsid之前,首先要了解两个概念:进程组和会话期  进程组:是一个或多个进程的集合。进程组有进程组ID来唯一标识。除了进程号(PID)之外,进程组ID也是一个进程的必备属性。每个进程组都有一个组长进程,其组长进程的进程号等于进程组ID。且该进程组ID不会因组长进程的退出而受到影响。  会话周期:会话期是一个或多个进程组的集合。通常,一个会话开始于用户登录,终止于用户退出,在此期间该用户运行的所有进程都属于这个会话期。  接下来就可以具体介绍setsid的相关内容:  (1)setsid函数作用:  setsid函数用于创建一个新的会话,并担任该会话组的组长。调用setsid有下面的3个作用:  让进程摆脱原会话的控制  让进程摆脱原进程组的控制  让进程摆脱原控制终端的控制  那么,在创建守护进程时为什么要调用setsid函数呢?由于创建守护进程的第一步调用了fork函数来创建子进程,再将父进程退出。由于在调用了fork函数时,子进程全盘拷贝了父进程的会话期、进程组、控制终端等,虽然父进程退出了,但会话期、进程组、控制终端等并没有改变,因此,这还不是真正意义上的独立开来,而setsid函数能够使进程完全独立出来,从而摆脱其他进程的控制。

改变当前目录为根目录

  这一步也是必要的步骤。使用fork创建的子进程继承了父进程的当前工作目录。由于在进程运行中,当前目录所在的文件系统(如“/mnt/usb”)是不能卸载的,这对以后的使用会造成诸多的麻烦(比如系统由于某种原因要进入单用户模式)。因此,通常的做法是让"/"作为守护进程的当前工作目录,这样就可以避免上述的问题,当然,如有特殊需要,也可以把当前工作目录换成其他的路径,如/tmp。改变工作目录的常见函数式chdir。

重设文件权限掩码

  文件权限掩码是指屏蔽掉文件权限中的对应位。比如,有个文件权限掩码是050,它就屏蔽了文件组拥有者的可读与可执行权限。由于使用fork函数新建的子进程继承了父进程的文件权限掩码,这就给该子进程使用文件带来了诸多的麻烦。因此,把文件权限掩码设置为0,可以大大增强该守护进程的灵活性。设置文件权限掩码的函数是umask。在这里,通常的使用方法为umask(0)。

关闭文件描述符

  同文件权限码一样,用fork函数新建的子进程会从父进程那里继承一些已经打开了的文件。这些被打开的文件可能永远不会被守护进程读写,但它们一样消耗系统资源,而且可能导致所在的文件系统无法卸下。  在上面的第二步之后,守护进程已经与所属的控制终端失去了联系。因此从终端输入的字符不可能达到守护进程,守护进程中用常规方法(如printf)输出的字符也不可能在终端上显示出来。所以,文件描述符为0、1和2 的3个文件(常说的输入、输出和报错)已经失去了存在的价值,也应被关闭。通常按如下方式关闭文件描述符:  ===============================  for(i=0;i<MAXFILE;i++)  close(i);  ===============================

守护进程退出处理

  当用户需要外部停止守护进程运行时,往往会使用 kill命令停止该守护进程。所以,守护进程中需要  编码来实现kill发出的signal信号处理,达到进程的正常退出。  ===============================  signal(SIGTERM, sigterm_handler);  void sigterm_handler(int arg)  {  _running = 0;  }  ===============================  这样,一个简单的守护进程就建立起来了。  实现守护进程的完整实例(每隔10s在/tmp/dameon.log中写入一句话):  =====================================================================  
#include<stdio.h>   
#include<stdlib.h>  
#include<string.h>   
#include<fcntl.h>  
#include<sys/types.h>   
#include<unistd.h>  
#include<sys/wait.h>  
#include <signal.h>  
#define MAXFILE 65535  
volatile sig_atomic_t _running = 1;  

int main()  
{  
    pid_t pc;  
    int i,fd,len;  
    char *buf="this is a Dameon\n";  
    len = strlen(buf);  
    pc = fork(); //第一步  
    if(pc<0)
    {  
        printf("error fork\n");  
        exit(1);  
    } 
   else if(PC>0) 
       exit(0);  
    setsid(); //第二步  
    chdir("/"); //第三步 
   umask(0); //第四步 
   for(i=0;i<MAXFILE;i++) //第五步 
   close(i);  
    signal(SIGTERM, sigterm_handler);  
    while( _running )
    {  
        if((fd=open("/tmp/dameon.log",O_CREAT|O_WRONLY|O_APPEND,0600))<0) 
        { 
           perror("open");  
            exit(1);  
        }  
        write(fd,buf,len);  
        close(fd);  
        usleep(10*1000); //10毫秒  
    }  
}  
void sigterm_handler(int arg) 
{ 
    _running = 0;  
}

linux守护进程列表

  amd:自动安装NFS(网络文件系统)守侯进程
  apmd:高级电源治理
  Arpwatch:记录日志并构建一个在LAN接口上看到的以太网地址和ip地址对数据库
  Autofs:自动安装治理进程automount,与NFS相关,依靠于NIS
  Bootparamd:引导参数服务器,为LAN上的无盘工作站提供引导所需的相关信息
  crond:linux下的计划任务
  Dhcpd:启动一个DHCP(动态IP地址分配)服务器
  Gated:网关路由守候进程,使用动态的OSPF路由选择协议
  Httpd:WEB服务器
  Inetd:支持多种网络服务的核心守候程序
  Innd:Usenet新闻服务器
  Linuxconf:答应使用本地WEB服务器作为用户接口来配置机器
  Lpd:打印服务器
  Mars-nwe:mars-nwe文件和用于Novell的打印服务器
  Mcserv:Midnight命令文件服务器
  named:DNS服务器
  netfs:安装NFS、Samba和NetWare网络文件系统
  network:激活已配置网络接口的脚本程序
  nfs:打开NFS服务
  nscd:nscd(Name Switch Cache daemon)服务器,用于NIS的一个支持服务,它高速缓存用户口令和组成成员关系
  portmap:RPC portmap治理器,与inetd类似,它治理基于RPC服务的连接
  postgresql:一种SQL数据库服务器
  routed:路由守候进程,使用动态RIP路由选择协议
  rstatd:一个为LAN上的其它机器收集和提供系统信息的守候程序
  ruserd:远程用户定位服务,这是一个基于RPC的服务,它提供关于当前记录到LAN上一个机器日志中的用户信息
  rwalld:激活rpc.rwall服务进程,这是一项基于RPC的服务,答应用户给每个注册到LAN机器上的其他终端写消息
  rwhod:激活rwhod服务进程,它支持LAN的rwho和ruptime服务
  sendmail:邮件服务器sendmail
  smb:Samba文件共享/打印服务
  snmpd:本地简单网络治理候进程
  squid:激活代理服务器squid
  syslog:一个让系统引导时起动syslog和klogd系统日志守候进程的脚本
  xfs:X Window字型服务器,为本地和远程X服务器提供字型集
  xntpd:网络时间服务器
  ypbind:为NIS(网络信息系统)客户机激活ypbind服务进程
  yppasswdd:NIS口令服务器
  ypserv:NIS主服务器
  gpm:管鼠标的
  identd:AUTH服务,在提供用户信息方面与finger类似    
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值