欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Unix环境高级编程笔记:13、守护进程

程序员文章站 2022-05-07 17:52:56
...
1 守护进程
    守护进程也称精灵进程(daemon)是生存期较长的一种进程。它们常常在系统自举时启动,仅在系统关闭时才终止。
    因为没的控制终端,所以它们是在后台运行的。
 
2、守护进程特征
    父进程ID为0的各进程通用是内核进程,它们作为系统自举过程中的一部分启动
    内核进程是特殊的,通用存在于系统的整个生命周期。它们以超级用户特权运行,无控制终端,无命令行。
 
    进程1通常是init,它是一个系统守护进程,负责启动各运行层次特定的系统服务。
 
    portmap网络端口映射服务
    syslogd守护进程把系统消息记入日志的任何程序使用
    xinted 侦听网络接口,取得来自网络的对各种网络服务进程的请求。  nfsd lockd rpciod
    cron
 
3、编程规则
    1)umask将文件模式创建屏蔽字设置为0    
    2)调用fork,然后使父进程退出
    3)调用setsid以创建一个新会话
    4) 将当前工作目录更改为根目录
    5)关闭不再需要的文件描述符
    6)某些守护进程打开/dev/null使其具有文件描述符0、1、2,这样任何一个试图读标准输入、写标准输出或标准出错的库例程都不会产生任何效果。
    
 
daemonize.c  想初始化成为一个守护进程的程序调用。    
#include <stdio.h>
#include <syslog.h>
#include <fcntl.h>
#include <stdlib.h>
#include <unistd.h>
 
void daemonize(const char *cmd) {
int i,fd0,fd1,fd2;
pid_t pid;
struct rlimit r1;
struct sigaction sa;
 
umask(0);
 
if(getrlimit(RLIMIT_NOFILE,&r1) < 0) {
printf("%s:can`t get file limit",cmd);
exit(1);
}
if(pid=fork()) < 0) {
printf("%s:can`t fork",cmd);
} else if(pid !=0) {
exit(0);
}
 
setsid();
sa.sa_handler = SIG_IGN;
sigemptyset(&sa.sa_mask);
sa.sa_flags = 0;
if(sigaction(SIGHUP,&sa,NULL) <0) {
printf("%s:can`t ignore SIGHUP");
exit(1)
}
if((pid = fork)) < 0) {
printf("%s:can`t fork")
exit(1);
}
else if(pid != 0) {
exit(0)
}
 
if(chdir("/") < 0) {
printf("%s:can`t change directory to /");
exit(1);
}
 
if(r1.rlim_max == RLIM_INFINITY)
r1.rlim_max = 1024;
for(i=0;i<r1.rlim_max;i++) {
close(i);
}
 
 
fd0 = open("/dev/null",O_RDWR);
fd1 = dup(0);
fd2 = dup(0);
 
openlog(cmd,LOG_CONS,LOG_DAEMON);
if(fd0 !=0 ||fd1 != 1||fd2!=2) {
syslog(LOG_ERR,"unexpected file descriptors %d %d %d",fd0,fd1,fd2);
exit(1);
}
}
 
#include <unistd.h>
int daemon(int nochdir,int noclose);
 
在创建精灵进程的时候,往往需要将精灵进程的工作目录修改为"/"根目录 
并且将标准输入,输出和错误输出重定向到/dev/null 

daemon的作用就是当参数nochdir为0时,将根目录修改为工作目录 
noclose为0时,做输入,输出以及错误输出重定向到/dev/null 
 
#include <unistd.h>
int main(int argc, char *argv[])
{
    ...
    if (daemon(0, 0)) {//调用glibc库函数daemon,创建daemon守护进程
        perror("daemon");
        return -1;
    }
    好了执行到这里的就是daemon的子进程了[luther.gliethttp].
    ...
}
 
4、出错记录
    三种方式产生日志消息
    1)内核例程可以调用log函数
    2)用户进程(守护进程)调用syslog(3)函数以产生日志消息
    3)在此主机上的一个用户进程
 
    #include <syslog.h>
    openlog
    syslog
    closelog
    setlogmask