00. 目录
01. 守护进程概述
守护进程(Daemon Process),也就是通常说的 Daemon 进程(精灵进程),是 Linux 中的后台服务进程。它是一个生存期较长的进程,通常独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。
守护进程是个特殊的孤儿进程,这种进程脱离终端,为什么要脱离终端呢?之所以脱离于终端是为了避免进程被任何终端所产生的信息所打断,其在执行过程中的信息也不在任何终端上显示。由于在 Linux 中,每一个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程都会依附于这个终端,这个终端就称为这些进程的控制终端,当控制终端被关闭时,相应的进程都会自动关闭。
Linux 的大多数服务器就是用守护进程实现的。比如,Internet 服务器 inetd,Web 服务器 httpd 等。
02. 守护进程查看方法
在终端输入ps axj
- a 表示不仅列当前用户的进程,也列出所有其他用户的进程
- x 表示不仅列有控制终端的进程,也列出所有无控制终端的进程
- j 表示列出与作业控制相关的信息
从上面的结果可以看出守护进程的一些特点:
- 守护进程基本上都是以超级用户启动( UID 为 0 )
- 没有控制终端( TTY 为 ?)
- 终端进程组 ID 为 -1 ( TPGID 表示终端进程组 ID)
一般情况下,守护进程可以通过以下方式启动:
- 在系统启动时由启动脚本启动,这些启动脚本通常放在 /etc/rc.d 目录下;
- 利用 inetd 超级服务器启动,如 telnet 等;
- 由 cron 定时启动以及在终端用 nohup 启动的进程也是守护进程。
03. 编写守护进程的步骤
3.1 屏蔽一些控制终端操作的信号
这是为了防止守护进行在没有运行起来前,控制终端受到干扰退出或挂起。
signal(SIGTTOU,SIG_IGN);
signal(SIGTTIN,SIG_IGN);
signal(SIGTSTP,SIG_IGN);
signal(SIGHUP ,SIG_IGN);
3.2 父进程退出
守护进程不可以是组长进程。方法是在进程中调用 fork() 使父进程终止, 让守护进行在子进程中后台执行。形式上脱离了控制终端。
if( pid = fork() ){ // 父进程
exit(0); //结束父进程,子进程继续
}
3.3 创建会话,完全脱离控制终端
有必要先介绍一下 Linux 中的进程与控制终端,登录会话和进程组之间的关系:进程属于一个进程组,进程组号(GID)就是进程组长的进程号(PID)。登录会话可以包含多个进程组。这些进程组共享一个控制终端。这个控制终端通常是创建进程的 shell 登录终端。 控制终端、登录会话和进程组通常是从父进程继承下来的。我们的目的就是要摆脱它们 ,使之不受它们的影响。因此需要调用 setsid() 使子进程成为新的会话组长,示例代码如下:
setsid();
setsid() 调用成功后,进程成为新的会话组长和新的进程组长,并与原来的登录会话和进程组脱离。由于会话过程对控制终端的独占性,进程同时与控制终端脱离。
3.4 关闭不需要文件描述符
进程从创建它的父进程那里继承了打开的文件描述符。如不关闭,将会浪费系统资源,造成进程所在的文件系统无法卸下以及引起无法预料的错误。按如下方法关闭它们:
// NOFILE 为 <sys/param.h> 的宏定义
// NOFILE 为文件描述符最大个数,不同系统有不同限制
for(i=0; i< NOFILE; ++i){// 关闭打开的文件描述符
close(i);
}
3.5 改变当前工作目录
进程活动时,其工作目录所在的文件系统不能卸下。一般需要将工作目录改变到根目录。对于需要转储核心,写运行日志的进程将工作目录改变到特定目录如 /tmp。示例代码如下:
chdir("/");
3.6 设置权限掩码
进程从创建它的父进程那里继承了文件创建掩码。它可能修改守护进程所创建的文件的存取权限。为防止这一点,将文件创建掩码清除:
umask(0);
3.7 处理 SIGCHLD 信号
但对于某些进程,特别是服务器进程往往在请求到来时生成子进程处理请求。如果父进程不等待子进程结束,子进程将成为僵尸进程(zombie)从而占用系统资源(关于僵尸进程的更多详情,请看《特殊进程之僵尸进程》)。如果父进程等待子进程结束,将增加父进程的负担,影响服务器进程的并发性能。在 Linux 下可以简单地将 SIGCHLD 信号的操作设为 SIG_IGN 。
signal(SIGCHLD, SIG_IGN);
这样,内核在子进程结束时不会产生僵尸进程。
04. 守护进程代码
守护进程参考代码:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
int main(void)
{
int i = 0;
pid_t pid = -1;
//a. 设置掩码
umask(0);
//b. 创建子进程 父进程退出
pid = fork();
if (-1 == pid)
{
perror("fork");
goto err0;
}
else if (0 != pid)
{
//父进程
exit(0);
}
//子进程
//c. 创建一个新的会话
setsid();
//d. 更改当前工作目录
chdir("/");
//e. 关系不需要的描述符
close(0);
close(1);
close(2);
//做事情
while(1)
{
system("echo hello world >> /tmp/a");
sleep(1);
}
return 0;
err0:
return 1;
}
测试结果:
05. 附录
5.1 参考博客:【Linux系统编程】特殊进程之守护进程