Linux _守护进程 浅解

守护进程

  1. 什么是守护进程
    守护进程,也称Daemon进程
    守护进程,是Linux的后台服务进程。

    普通进程和终端的关系:
    用户与系统交流的界面,称为“终端”。
    当在某个终端上运行某个进程时,该终端就称为该进程的“控制终端”。
    当控制终端关闭时,它对应的进程(在该终端上启动的进程)都将被自动关闭。

    守护进程和终端的关系:
    守护进程不依赖于任何终端。
    守护进程启动后,将一直运行到系统关闭.

  2. 什么时候使用守护进程
    如果希望某个进程不要受到用户、终端的影响,则可以把该进程实现为“守护进程”。

  3. 守护进程的编写
    步骤:
    1) 使该进程脱离控制台
    (1) 创建一个子进程,
    接着关闭该子进程的父进程。
    使得该子进程变成孤儿进程,进而被init收养,称为init的子进程。
    即:
    pd = fork();
    if (pd > 0) {
    exit(0);
    }

    (2) 在子进程中创建新会话
         即:
             setsid();
             作用:使该进程脱离原控制终端的控制。
                     使该进程脱离原进程组的控制。
                     使该进程脱离原会话的控制。
    
        补充:
        进程组: 进程组,就是一个、或多个进程的集合。
                    每个进程组,都有一个“进程组ID”
                    每个进程组,都有一个组长进程,其组长进程进程的PID就是进程组的“进程组ID”
    
        会话:一个会话是一个、或多个进程组的集合。
                一个会话,开始于用户登录,终止于该用户退出。
    

    2)改变该进程的当前目录。
    该进程的当前目录继承自父进程。
    把该进程的当前目录改为根目录。
    方法:chdir();

    3) 改变文件权限掩码
    该进程的文件权限掩码,继承自父进程。
    方法:umask()
    一般使用umask(0), 即放开所有屏蔽。

    补充:
    文件权限掩码:
    不是指文件的访问权限。
    例:如果某文件的文件权限掩码是050,
        则表示,该文件的文件组拥有者,没有读权限、没有写权限。
    

    4)关闭文件描述符
    该进程的文件描述符,继承自其父进程。
    而,守护进程已不可能在终端上输出、输入,所以文件描述符0,1,2都不会再用。
    其他已继承下来的文件描述符也不再使用。
    所以,需要关闭文件描述符,以节省资源。
    即:
    for (i=0; i

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <syslog.h>
#include <sys/klog.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define FILE_NAME "test.txt"
#define BUFF_SIZE  80

int main(void)
{
    pid_t pd;
    int i;
    int fd_max;
    int fd;
    time_t cur_time;
    char buff[BUFF_SIZE];
    int ret;

    pd = fork();
    if (pd == -1) {
        printf("fork failed!\n");  //¥À ±printfªπø…“‘ π”√
        exit(1);        
    } else if (pd > 0) {
        exit(0);
    }

    setsid();
    chdir("/");
    umask(0);
    fd_max = getdtablesize();
    for (i=0; i<fd_max; i++) {
        close(i);
    }

       //÷¡¥À£¨ ÿª§Ω¯≥Ã¥¥Ω®Õͱœ

    // ¥Úø™syslog∑˛ŒÒ
    openlog("my daemon", LOG_PID, LOG_DAEMON);
    fd = open(FILE_NAME, O_WRONLY | O_CREAT, 0666);
    //fd = open(FILE_NAME, O_WRONLY);
    if (fd == -1) {
        syslog(LOG_ERR, "open file %s failed!\n", FILE_NAME);
        exit(1);
    }

    syslog(LOG_INFO, "open file %s succeed!\n", FILE_NAME);

    while(1) {
        time(&cur_time);
        sprintf(buff, "%s", ctime(&cur_time));
        ret = write(fd, buff, strlen(buff) + 1);
        if (ret == -1) {
            syslog(LOG_ERR, "write file %s failed!\n", FILE_NAME);
            exit(1);
        }
        sleep(5);
    }

    closelog();
       return 0;    
}

posted @ 2016-04-01 13:34  夜色下的港湾  Views(158)  Comments(0Edit  收藏  举报