守护进程C实现简单分析
2013-03-25 10:17
411 查看
守护进程
记录在这里给自己一个备份,至于具体代码则不一一分析。
特点:
守护进程(Daemon)是在后台运行的一种特殊进程,脱离于终端。因为这可以避免进程被任何终端所产生的信息所打断,它在执行过程中产生的信息也不在终端上显示。
守护进程周期性的执行某种任务或等待处理某些发生的事件,Linux的大多服务器就是使用守护进程实现的。eg:Internet服务器inetd,和Web服务器httpd等。
启动方式:
1.在系统启动时由启动脚本启动,通常放置在/etc/rc.d目录下。
2.利用inetd超级服务器启动,eg:telnet..
3.由cron定时启动以及在终端用nohup启动的进程也是守护进程。
编程过程:
1. 屏蔽一些有关控制终端操作的信号。为防止守护进程在没有正常运行起来前,控制终端收到干扰退出或挂起。
eg: signal(SIGTTOU, SIG_IGN)
signal(SIGTTIN, SIG_IGN)
signal(SIGTSTP, SIG_IGN)
signal(SIGHUP, SIG_IGN)
2. 在后台运行。这是为了避免挂起控制终端将其放入后台执行。方法是在进程中调用fork使父进程终止,让其在子进程中后台执行。
eg: if(pid = fork())
exit(0); //父进程结束,子进程继续
3. 脱离控制终端与进程组。因为进程属于一个进程组,进程组号(PGID)就是进程组长的进程号(PID)。同进程组中的进程共享一个控制终端,这个控制终端通常是创建进程的shell登陆终端。
而控制终端和进程组通常是从父进程继承下来的。需要摆脱他们,使之不受他们的影响。因此需要调用setid()使子进程成为新的会话组长。
eg: setsid()
调用成功后,调用进程会成为新的会话组长和进程组长,并与原来的登陆会话和进程组脱离。由于会话过程对中断的独占性,进程同时与终端脱离。
4. 禁止进程重新打开控制终端。现在,进程已经成为无终端的会话组长。但它可以重新申请打开一个控制终端。可以通过使进程不再成为会话组长来禁止进程重新打开控制终端,采用的办法是再次穿件一个子进程。
eg: if(pid = fork())
exit(0);
5. 关闭打开的文件描述符。进程从创建它的父进程那里继承了打开的文件描述符,一般情况下,不再需要,包括标准输入输出(因为守护进程是后台执行)。
如不关闭,将会浪费系统资源,造成进程所在的文件系统无法卸下并引起无法预料的错误。按以下方法关闭:
eg: #define NOFILE 256 //不同的系统有不同的限制
for(i = 0; i < NOFILE; i++)
close(i); //关闭打开的文件描述符
6. 改变当前工作目录。进程活动时,其工作目录所在的文件系统不能卸下。因此,一般需要守护进程的工作目录改变到其根目录。写运行日志的进程将工作兖改变到特定的目录如/tmp。
eg: chdir("/");
7. 重设文件创建掩模。进程从创建它的父进程那里继承了文件创建掩模。它可能修改守护进程所创建的文件的存取权限。为防止这一点,将文件创建掩模清除。
eg: umask(0);
8. 处理SIGCHLD信号(子进程退出信号)。但对于某些进程,特别是服务器进程,往往在请求到来时生成子进程处理请求。如果父进程不等待子进程结束,子进程将成为僵尸进程(zombie)从而占用系统资源。
如果父进程等待子进程结束,将增加父进程的负担,影响服务器进程的并发性能。在Linux下可以简单地将SIGCHLD信号的炒作设为SIG_IGN来解决这一问题。
eg: signal(SIGCHLD, SIG_IGN);
记录在这里给自己一个备份,至于具体代码则不一一分析。
特点:
守护进程(Daemon)是在后台运行的一种特殊进程,脱离于终端。因为这可以避免进程被任何终端所产生的信息所打断,它在执行过程中产生的信息也不在终端上显示。
守护进程周期性的执行某种任务或等待处理某些发生的事件,Linux的大多服务器就是使用守护进程实现的。eg:Internet服务器inetd,和Web服务器httpd等。
启动方式:
1.在系统启动时由启动脚本启动,通常放置在/etc/rc.d目录下。
2.利用inetd超级服务器启动,eg:telnet..
3.由cron定时启动以及在终端用nohup启动的进程也是守护进程。
编程过程:
1. 屏蔽一些有关控制终端操作的信号。为防止守护进程在没有正常运行起来前,控制终端收到干扰退出或挂起。
eg: signal(SIGTTOU, SIG_IGN)
signal(SIGTTIN, SIG_IGN)
signal(SIGTSTP, SIG_IGN)
signal(SIGHUP, SIG_IGN)
2. 在后台运行。这是为了避免挂起控制终端将其放入后台执行。方法是在进程中调用fork使父进程终止,让其在子进程中后台执行。
eg: if(pid = fork())
exit(0); //父进程结束,子进程继续
3. 脱离控制终端与进程组。因为进程属于一个进程组,进程组号(PGID)就是进程组长的进程号(PID)。同进程组中的进程共享一个控制终端,这个控制终端通常是创建进程的shell登陆终端。
而控制终端和进程组通常是从父进程继承下来的。需要摆脱他们,使之不受他们的影响。因此需要调用setid()使子进程成为新的会话组长。
eg: setsid()
调用成功后,调用进程会成为新的会话组长和进程组长,并与原来的登陆会话和进程组脱离。由于会话过程对中断的独占性,进程同时与终端脱离。
4. 禁止进程重新打开控制终端。现在,进程已经成为无终端的会话组长。但它可以重新申请打开一个控制终端。可以通过使进程不再成为会话组长来禁止进程重新打开控制终端,采用的办法是再次穿件一个子进程。
eg: if(pid = fork())
exit(0);
5. 关闭打开的文件描述符。进程从创建它的父进程那里继承了打开的文件描述符,一般情况下,不再需要,包括标准输入输出(因为守护进程是后台执行)。
如不关闭,将会浪费系统资源,造成进程所在的文件系统无法卸下并引起无法预料的错误。按以下方法关闭:
eg: #define NOFILE 256 //不同的系统有不同的限制
for(i = 0; i < NOFILE; i++)
close(i); //关闭打开的文件描述符
6. 改变当前工作目录。进程活动时,其工作目录所在的文件系统不能卸下。因此,一般需要守护进程的工作目录改变到其根目录。写运行日志的进程将工作兖改变到特定的目录如/tmp。
eg: chdir("/");
7. 重设文件创建掩模。进程从创建它的父进程那里继承了文件创建掩模。它可能修改守护进程所创建的文件的存取权限。为防止这一点,将文件创建掩模清除。
eg: umask(0);
8. 处理SIGCHLD信号(子进程退出信号)。但对于某些进程,特别是服务器进程,往往在请求到来时生成子进程处理请求。如果父进程不等待子进程结束,子进程将成为僵尸进程(zombie)从而占用系统资源。
如果父进程等待子进程结束,将增加父进程的负担,影响服务器进程的并发性能。在Linux下可以简单地将SIGCHLD信号的炒作设为SIG_IGN来解决这一问题。
eg: signal(SIGCHLD, SIG_IGN);
相关文章推荐
- Linux下一个简单守护进程的实现 (Daemon)
- APP中一种在Java层实现的简单守护进程方式
- 守护进程简单实现
- 简单实现一个守护进程(Daemon)
- Linux下一个简单守护进程的实现 (Daemon)
- 实现简单的ring3进程保护、hook OpenProcess函数、简单分析
- 介绍并实现简单的守护进程
- 基于linux 守护进程的分析与实现
- Win32 守护进程实现-- ACL 源码分析
- 守护进程(简单实现)--程序关闭后自动启动
- 【转】Windows守护进程的一种简单实现
- 一个简单进程调度器的实现和分析
- php实现简单的守护进程
- Windows守护进程的一种简单实现
- APP中一种在Java层实现的简单守护进程方式
- C语言实现简单的守护进程及信号处理
- Linux下一个简单守护进程的实现 (Daemon)
- 守护进程的最简单实现方法
- APP中一种在Java层实现的简单守护进程方式
- 基于Linux守护进程的分析与实现