daemon进程
daemon进程创建步骤
mydaemon.c
#include <stdio.h> #include <stdlib.h> #include <time.h> #include <sys/stat.h> #include <unistd.h> #include <sys/types.h> #include <errno.h> #include <signal.h> #include <fcntl.h> #define ERROR(flag) \ if(flag) \ { \ printf("%d: ",__LINE__); \ fflush(stdout); \ perror("error"); \ exit(errno); \ } int init_daemon() { pid_t pid = fork(); ERROR(pid == -1); if(pid > 0) exit(0); printf("pid = %d\n",getpid());
//-------------------------------------------------------- int ret = setsid(); ERROR(ret == -1);
//下面的创建新进程后,父退子留是system V的建议, 以保证不是会话的首进程, 防止取得控制终端 pid = fork(); ERROR(pid == -1); if(pid > 0) exit(0); //-------------------------------------------------------- close(STDIN_FILENO); close(STDOUT_FILENO); close(STDERR_FILENO); //-------------------------------------------------------- chdir("/"); //-------------------------------------------------------- umask(0); //-------------------------------------------------------- signal(SIGCHLD,SIG_IGN); } int main(int argc, char *argv[]) { time_t now; init_daemon(); while(1) { int fd = open("daemon.log",O_WRONLY | O_CREAT | O_APPEND, 0600); time(&now); char *s = ctime(&now); write(fd,s,strlen(s)); sleep(2); close(fd); } #if 0 syslog(LOG_USER | LOG_INFO,"test daemon, pid = %d\n",getpid()); while(1) { sleep(3); time(&now); syslog(LOG_USER | LOG_INFO,"hello now is: %s\n",ctime(&now)); } #endif return 0; }
编译链接执行, 输出结果如下:
查询进程相关信息结果如下:
daemon进程的规则在init_daemon()函数中得到满足:
1. 调用fork()后父进程退出. 使得shell认为这条命令已完毕, 子进程不是一个进程组的组长.
2. 创建新会话(通过fork()创建新进程后,关闭父进程来保证成功). 该进程将没有控制终端
3. 关闭不需要的文件描述符
4. 当前工作目录改为根目录
5. 文件模式创建屏蔽字设置为0
6. 忽略SIGCHLD信号(似乎不必)