编写Linux系统下Daemon程序的方法步骤

来源:互联网 发布:三星手机3g网络设置 编辑:程序博客网 时间:2024/05/19 09:11

转自:http://blog.csdn.net/analogous_love/article/details/52806722

一、引言 Daemon程序是一直运行的服务端程序,又称为守护进程。 
本文介绍了在Linux下编写Daemon程序的步骤,并给出了例子程序。 
二、Daemon程序简介 Daemon是长时间运行的进程,通常在系统启动后就运行,在系统关闭时才结束。一般说Daemon程序在后台运行,是因为它没有控制终端,无法和前台的用户交互。Daemon程序一般都作为服务程序使用,等待客户端程序与它通信。我们也把运行的Daemon程序称作守护进程。 
三、Daemon程序编写规则 
编写Daemon程序有一些基本的规则,以避免不必要的麻烦。 
1、首先是程序运行后调用fork,并让父进程退出。子进程获得一个新的进程ID,但继承了父进程的进程组ID。 
2、调用setsid创建一个新的session,使自己成为新session和新进程组的leader,并使进程没有控制终端(tty)。 
3、改变当前工作目录至根目录,以免影响可加载文件系统。或者也可以改变到某些特定的目录。 
4、设置文件创建mask为0,避免创建文件时权限的影响。 
5、关闭不需要的打开文件描述符。因为Daemon程序在后台执行,不需要于终端交互,通常就关闭STDIN、STDOUT和STDERR。其它根据实际情况处理。

另一个问题是Daemon程序不能和终端交互,也就无法使用printf方法输出信息了。我们可以使用syslog机制来实现信息的输出,方便程序的调试。在使用syslog前需要首先启动syslogd程序,关于syslogd程序的使用请参考它的man page,或相关文档,我们就不在这里讨论了。 
四、一个Daemon程序的例子 编译运行环境为Redhat Linux 8.0。 
我们新建一个daemontest.c程序,文件内容如下:

int daemon_init(void){    pid_t pid;    if ((pid = fork()) < 0)         return(-1);    else if (pid != 0)         exit(0); /* parent exit */    /* child continues */    setsid(); /* become session leader */    chdir("/"); /* change working directory */    umask(0); /* clear file mode creation mask */    close(0); /* close stdin */    close(1); /* close stdout */    close(2); /* close stderr */    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);    }}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25

在实际项目中我是这么做的:

if (isdaemon) {        int pid;        signal(SIGCHLD, SIG_IGN);        //1)在父进程中,fork返回新创建子进程的进程ID;        //2)在子进程中,fork返回0;        //3)如果出现错误,fork返回一个负值;        pid = fork();        if (pid < 0) {            perror("fork");            exit(-1);        }        //父进程退出,子进程独立运行        else if (pid > 0) {            exit(0);        }        //之前parent和child运行在同一个session里,parent是会话(session)的领头进程,        //parent进程作为会话的领头进程,如果exit结束执行的话,那么子进程会成为孤儿进程,并被init收养。        //执行setsid()之后,child将重新获得一个新的会话(session)id。        //这时parent退出之后,将不会影响到child了。        setsid();        int fd;        fd = open("/dev/null",O_RDWR,0);        if (fd != -1) {            dup2(fd,STDIN_FILENO);            dup2(fd,STDOUT_FILENO);            dup2(fd,STDERR_FILENO);        }        if (fd > 2)            close(fd);    }

原创粉丝点击