下面的完成了这样一个功能,创建一个守护进程,每个一秒在/tmp目录下的文件peng.txt中记录当前系统时间。

 

一、守护进程

守护进程是linux中的后台服务进程,在系统启动时开始运行,在系统关闭时终止。Linux系统中的大多数服务进程都是由守护进程实现的。

 

二、创建守护进程

 

  • 创建子进程,父进程退出

此时,子进程变成孤儿进程,在后台运行。此时,子进程表面上脱离了终端(如bash),但是实际上却没有,因为其PGID和SID都是原来的没变。

  • 在子进程中创建新的会话

调用setsid()函数,这样,子进程的GID和sid就都等于子进程pid了,它使子进程完全独立出来,可以使用 kill –9 pid,将其杀死。

  • 改变进程的工作目录为根目录

首先要知道工作目录与当前目录的区别,工作目录默认是当前目录,比如执行函数open(“1.c”,……),时,系统会自动到默认的工作目录(即当前目录)中读取或者创建文件 1.c ,但是通过调用函数 chdir(“/tmp”),可以将工作目录变成/tmp,此后,如果再执行open(“1.c”,……),系统就回到/tmp目录下面读取或者创建文件 1.c;

其次,为什么有改变工作目录?原因是,守护进程的工作目录不可以被卸载!假如你把U盘挂载到了机器上,你没改工作目录,U盘上有一个守护进程的程序,你运行了,那么U盘所挂载的目录就是该守护进程的工作目录,假如守护进程中有写文件和读文件操作,你的U盘就没法卸载了,假如你把工作目录改成/tmp,而/tmp无法卸载,运行U盘上的守护进程后,再与U盘无关,此时你可以卸载U盘,而不会对该守护进程有什么影响,该守护进程已经在内存中了。

通常将守护进程的工作目录设置成 “/”或者“/tmp ”,可以看看/tmp 目录的权限是否够用,一般是 0777。

  • 重设文件权限掩码

将文件权限掩码设置为umask(0),这样可以增加守护进程的灵活性

  • 关闭文件描述符

因为创建子进程时,父进程已经打开了若干文件(比如stdin、stdout、stderr),而守护进程用不到这些文件,所以应该关闭。因此,守护进程已经无法使用诸如printf 、scanf、gets等等。

 

源代码如下:



#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <time.h>

int main()
{
    pid_t pid;
    int i;
    FILE *fp;
    time_t t;

    if ((pid = fork()) == -1)  //创建子进程
    {
        perror("fork");
        exit(-1);
    }

    if (pid > 0) exit(0);   //pid>0是父进程,父进程退出

    setsid();         //在子进程中建立新会话,使子进程完全独立
    chdir ("/tmp");   //改变工作目录为/tmp
    umask(0);        //重设文件权限掩码
    for (i = 0; i < getdtablesize(); i++)   //关闭已经打开的文件,包括stdin、stdout和stderr,getdtablesize的返回值为固定值
        close(i);                            //注意getdtablesize()不是函数,而是一个宏定义,为常数。

    if ((fp = fopen("time.txt", "a")) == NULL)  //打开要写的文件
        exit(-1);

    while (1)
    {
        sleep(1);
        t = time(NULL);  //获取系统时间
        fprintf(fp, "%s", ctime(&t));//ctime将time转化为标准格式字符串,返回字符串首地址
        fflush(fp);  //因为是全缓存,所以为保证实时性,用该用fflush进行刷新。
    }

    return 0;
}