Linux 守护进程

守护进程
守护进程(daemon)是生存期长的一种进程。它们常常在系统引导装入时起动
,在系统关闭时终止。因为它们没有控制终端,所以说它们是在后台运行的。Linux系统有很多守护进程,它们执行日常事物活动。(带d的都是守护进程。)

守护进程特征:
(1)所有守护进程都以超级用户(用户ID为0)的优先权运行。
(2)没有一个守护进程具有控制终端——终端名称设置为问号(?)、终端
前台进程组ID设置为——1。缺少控制终端可能是精灵进程调用setsid的结果
(3)除upate以外的所有精灵进程都是进程组的首进程,对话期的首进程,而
且是这些进程组和对话期中的唯一进程。update是它所在进程组和对话期(中的唯一进程,但是该进程组的首进程(可能也是该对话期的首进程)已经终止
(4)所有这些守护进程的父进程都是init进程。

守护进程编程规则(5步)
(1)创建子进程,父进程退出:首先做的是调用fork,然后使父进程exit。
这样做实现了下面几点:第一,如果该守护进程由一条简单SHELL命令起动的,那么使父进程终止使得SHELL认为这条命令已经执行完成。第二,子进程继承了父进程的进程组ID,但具有一个新的进程ID,这就保证了子进程不是一个进程组的首进程。这对于下面就要做的是setsid调用是必要的前提条件。
(2)调用setsid以创建一个新的会话,并担任该会话组的组长,调用setsid
作用有三个:A成为新对话期的首进程。B、成为一个新进程组的首进程。C、脱离控制终端。(会话组是一个或多个进程组的集合)。
setsid()函数格式:
#include<sys/types.h>
#include<unist.h>
Pid_t setsid(void)
函数成功时返回该进程的ID号,出错时返回-1。
(3)改变当前目录为根目录——chdir("/");从父进程继承过来的当前工作目
录可能在一个mnt的文件系统中。因为守护进程能常在系统再引导之前是一直存在的,所以如果守护进程的当前工作目录在一个mnt文件系统中,那么该文件系统就不能被拆卸。
(4)重设文件权限掩码——umask(0);由继承得来的文件方式创建屏蔽定可能
会拒绝设置某些许可权。例如,若守护进程要创建一个组可读、写的文件,而继承的文件方式创建屏蔽字,屏蔽了这两种许可权,则所要求的组可读、写就不能起作用。
(5)关闭不再需要的文件描述符。用fork函数创建的子程序会从父进程那继
承一些已经打开的文件,由此为使守护进程就不再持有从其父进程继承来的某些文件描述符。但是,究竟关闭哪些描述符则与具体的精灵进程有关,可以程序中的方法关闭所有文件描述符。
for(i=0;i<MAXFILE;i++)
close(i);
守护例如:dameon.c
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<fcntl.h>
#inlcude<sys/types.h>
#inlcude<unistd.h>
#include<sys/wait.h>
#define MAXFILE 65535

int main()
{
pid_t pc;
int i,fd,len;
char *buf="This is a Dameon/n";
len=strlen(buf);
pc=fork();
if(pc<0){
printf("error fork/n");
exit(1);
}
else if(pc>0)
exit(0);
setid();
chdir("/");
umask(0);
for(i=0;i<MAXFILE;i++)
close(i);
while(1){
if((fd=open("/tmp/dameon.log",O_CREAT|O_WRONLY|O_APPEND,0600))<0){
perror("open");
exit(1);
}
write(fd,buf,len+1);
close(fd);
sleep(10);
}
}

守护进程的出错处理
由于守护进程完全脱离了控制终端,因此,不能像其他程序一样通过输出错误
信息到控制台的方式来通知程序员。通常的办法是使用syslog服务,将出错信息输入到“/var/log/message“系统日志文件中去syslog是linux中的系统日志管理服务通过守护进程syslog来维护。
syslog函数说明
Openlog函数用于打开系统日志服务的一个连接;
Syslog函数用于向日志文件中写入消息,在这里可以规定消息的优先级、消息
的输出格式等。
Closelog函数用于关闭系统日志服务的连接。
(1)openlog函数
#include<syslog.h>
void openlog(char* ident,int option,int facility);
Ident:要向每个消息加入的字符串,通常为程序的名称;
option参数:
LOG_CONS:若日志消息不能通过发送至syslogd,则将该消息写至控制台;
LOG_NDELAY:立即打开UNIX域数据报套接口至syslsgd守护进程。通常,在记录

第一条消息之前,该套接口不打开。
LOG_PERROR:除将日志消息发送给syslog外,还将它写至标准出错(stderr)。
LOG_PID:每条消息都包含进程ID,此选择项可供对每个请求都fork一个子进程
的守护进程使用。
openlog的facility参数;
LOG_AUTH授权程序:login.su,getty,...
LOG_CRON cron和at
LOG_DAEMOMN系统守护进程:ftpd,routed,....
LOG_KERN内核产生的消息
LOG_KOCALO~7保留由本地使用
LOG_LPR行打系统:lpd,lpc,....
LOG_MAIL邮件系统
LOG_NEWSU senet网络新闻系统
LOG_SYSLOG syslogd守护进程本身
LOG_USER 来自其他用户进程的消息。
LOG_UUCP UUCP系统

(2)syslog函数
#include<syslog.h>
void syslog(int priority, char * format,...);
Priority选项(消息优先级)
LOG_EMERG紧急(系统不可使用)(最高优先级)
LOG_ALERT必须立即修复的条件
LOG_CRIT临界条件(例如,硬设备出错)
LOG_ERR出错条件
LOG_WARNING警告条件
LOG_NOTICE正常,但重要的条件
LOG_INFO信息性消息
LOG_DEBUG调试排错消息(最低优先级)

(3)closelog函数
#include<syslog.h>
void closelog();

syslog_dameon.c
#include<stdio.h>
#include<stdlib.h>
#include<stirng.h>
#include<fcntl.h>
#inlcude<sys/types.h>
#include<unistd.h>
#include<sys/wait.h>
#include<syslog.h>
#define MAXFILE 65535
int main()
{
pid_t pc,sid;
int i,fd,len;
char *buf="This is a Dameon/n";
len=strlen(buf);
pc=fork();
if(pc<0){
printf("error fork/n");
exit(1);
}else if(pc>0)
exit(0);
openlog("demo_update",LOG_PID,LOG_DAEMON);
if((sid=setsid())<0){
syslog(LOG_ERR,"%s/n","setsid");
exit(1);
}
if((sid=chdir("/"))<0){
syslog(LOG_ERR,"%s/n","chdir");
exit(1);
}
umask(0);
for(i=0;i<MAXFILE;i++)
close(i);
while(1){
if((fd=open("/tmp/dameon.log",O_CREATE|O_WRONLY|O_APPEND,0600))<0)
{syslog(LOG_ERR,"open");
exit(1);
}
write(fd,buf,len+1);
close(fd);
sleep(10);
}
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值