什么是守护进程?

守护进程是生存期长的一种进程,它们常常在系统引导装入时启动,仅在系统关闭时在终止。它们没有控制终端并且在后台运行。Linux

系统中有很多守护进程用以执行系统的日常事物,而且服务器程序一般都作为守护进程运行。大多数守护进程都以超级用户特权运行,而

且用户层守护进程的父进程是init进程。

如果你想查看一下你系统中有哪些守护进程,可以在终端下输入ps -x命令查看,TTY?的列说明此进程没有控制终端,即守护进程。

一些必要了解的基本概念

进程组:进程组是一个或多个进程的集合。每个进程都属于一个进程组,同一个进程组的各进程接受来自同一终端的各种信号。每个进程组

都有一个唯一的进程组ID。每个进程组都有一个组长进程,组长进程的进程组ID等于其进程ID。

会话:会话是一个或多个进程组的集合。会话通常是由shell的管道将几个进程编成一组的。

例如:在shell下输入如下命令:

program1 | program2 &

program3 | program4

则在此刻shell中包括3个进程组:登录shell组,由program1和program2组成的进程组,由program3和program4组成的进程组。前两个进程组

是后台进程组,第三个进程组为前台进程组。这三个进程组组成一个会话。

可调用setsid函数建立一个新会话,如果调用这个函数的进程已经是一个进程组的组长进程,则此函数返回出错。

setsid函数会做以下三件事:

  1. 时调用进程变成新会话的会话首进程,此时该进程是新会话的唯一进程。

  2. 使调用进程成为一个新进程组的组长进程,新进程组的组ID是该调用进程的进程ID。

  3. 调用进程没有控制终端。

创建守护进程的步骤

启动任意一个程序并让它成为守护进程运行需要以下步骤:

1、 用umask将文件模式创建屏蔽字设为一个已知值,由继承得来的文件模式创建屏蔽字可能会被设置为拒绝某些权限。如果守护进程要创建

文件可能要设置特定的权限。

2、 调用fork然后使父进程exit。这样父进程终止,而子进程则变为后台进程(也为孤儿进程)。虽然子进程继承父进程的进程组ID,但子进

程获得了一个新的进程ID,这就保证了子进程不是一个进程组的组长进程,这为子进程调用setsid函数创造条件。

3、 调用setsid创建一个新的会话,1)使调用进程成为新会话的首进程。2)成为一个新进程组的组长。3)没有控制终端。

4、 再次调用fork并终止父进程,继续使用子进程(第二个子进程)中的守护进程。这就保证了该守护进程不是会话首进程从而可以防止它取

得控制终端。因为当会话首进程打开第一个尚未与一个会话相关联的终端设备时,如果在调用open函数时没有指定O_NOCTTY标志,系统会将

此控制终端分配给会话,这样就不满足守护进程无控制终端的要求了。

5、 忽略SIGHUP信号,以为当会话首进程(第一个fork产生的第一个子进程)终止时,会话中的所有进程(在此指第二个子进程)都会收到SIGHUP

信号。

6、 将当前工作目录改为根目录以防止守护进程当前的工作目录在一个挂载的文件系统中。

7、 关闭所有打开的文件描述符并将标准输入,标准输出和标准错误输出重定向到/dev/null中。这样使任何一个试图读标准输入,写标准输出或

标准错误的库函数都不会产生任何效果。

8、 由于守护进程没有控制终端,那么它该如何输出出错消息呢。答案是将错误消息写到日志文件中去,调用syslog函数写出错消息。

实现代码

《UNIX环境高级编程 第三版》和《UNIX网络编程 卷一 第三版》都给出了实现代码,不过我认为前者更好。稍加修改后的代码如下:

void daemonize(const char *cmd)   // 参数cmd是程序的名字
{
int i, fd0, fd1, fd2;
pid_t pid;
struct rlimit rl;
struct sigaction sa; // 设置文件创建屏蔽字
umask(0); // getrlimit函数用于获取进程可用的最大的文件描述符
if(getrlimit(RLIMIT_NOFILE, &rl) < 0)
{
fprintf(stderr, "%s: can't get file limit", cmd);
exit(1);
} // 成为会话首进程,失去控制终端
if((pid = fork()) < 0)
{
fprintf(stderr, "%s: can't fork", cmd);
exit(1);
}
else if(pid != 0)
exit(0); // 终止父进程
setsid(); // 忽略产生的SIGHUP信号
sa.sa_handler = SIG_IGN;
sigemptyset(&sa.sa_mask);
sa.sa_flags = 0;
if(sigaction(SIGHUP, &sa, NULL) < 0)
{
fprintf(stderr, "%s: can't ignore SIGHUP", cmd);
exit(1);
}
if((pid = fork()) < 0)
{
fprintf(stderr, "%s: can't fork", cmd);
exit(1);
}
else if(pid != 0)
exit(0); // 终止父进程 // 改变当前工作目录为根目录
if(chdir("/") < 0)
{
fprintf(stderr, "%s: can't change directory to /", cmd);
exit(1);
} // 关闭所有文件描述符
if(rl.rlim_max == RLIM_INFINITY)
rl.rlim_max = 1024; // 如果进程可用的文件描述符为无限制则设为1024
for(i = 0; i < rl.rlim_max; ++i)
close(i); // 将标准输入,标准输出和标准错误输出重定向到/dev/null
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);
}
}

Reference:

《UNIX环境高级编程 第三版》

《UNIX网络编程 卷一 第三版》

Linux守护进程的更多相关文章

  1. .NET跨平台实践:用C#开发Linux守护进程

    Linux守护进程(Daemon)是Linux的后台服务进程,它脱离了与控制终端的关联,直接由Linux init进程管理其生命周期,即使你关闭了控制台,daemon也能在后台正常工作. 一句话,为L ...

  2. .NET跨平台实践:用C#开发Linux守护进程(转)

    Linux守护进程(Daemon)是Linux的后台服务进程,它脱离了与控制终端的关联,直接由Linux init进程管理其生命周期,即使你关闭了控制台,daemon也能在后台正常工作. 一句话,为L ...

  3. [转]❲阮一峰❳Linux 守护进程的启动方法

    ❲阮一峰❳Linux 守护进程的启动方法 "守护进程"(daemon)就是一直在后台运行的进程(daemon). 本文介绍如何将一个 Web 应用,启动为守护进程. 一.问题的由来 ...

  4. Server Develop (七) Linux 守护进程

    守护进程 守护进程,也就是通常说的Daemon进程,是Linux中的后台服务进程.它是一个生存期较长的进程,通常独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件.守护进程常常在系统引导装 ...

  5. Linux 守护进程和超级守护进程(xinetd)

    一 .Linux守护进程 Linux 服务器在启动时需要启动很多系统服务,它们向本地和网络用户提供了Linux的系统功能接口,直接面向应用程序和用户.提供这些服务的程序是由运行在后台的守护进程来执行的 ...

  6. Linux守护进程详解(init.d和xinetd) [转]

    一 Linux守护进程 Linux 服务器在启动时需要启动很多系统服务,它们向本地和网络用户提供了Linux的系统功能接口,直接面向应用程序和用户.提供这些服务的程序是由运行在后台 的守护进程来执行的 ...

  7. Linux守护进程的编程实现

    Linux 守护进程的编程方法 守护进程(Daemon)是执行在后台的一种特殊进程.它独立于控制终端而且周期性地执行某种任务或等待处理某些发生的事件.守护进程是一种非常实用的进程.Linux的大多数s ...

  8. Linux守护进程详解(init.d和xinetd)

    一 Linux守护进程 Linux 服务器在启动时需要启动很多系统服务,它们向本地和网络用户提供了Linux的系统功能接口,直接面向应用程序和用户.提供这些服务的程序是由运行在后台的守护进程来执行的. ...

  9. C#开发Linux守护进程

    用C#开发Linux守护进程   Linux守护进程(Daemon)是Linux的后台服务进程,它脱离了与控制终端的关联,直接由Linux init进程管理其生命周期,即使你关闭了控制台,daemon ...

  10. 笔记整理--Linux守护进程

    Linux多进程开发(三)进程创建之守护进程的学习 - _Liang_Happy_Life__Dream - 51CTO技术博客 - Google Chrome (2013/10/11 16:48:2 ...

随机推荐

  1. Mysql 客户端查询结果如何保存到本地而不是服务端?

    应用场景:知道某台DB服务器的IP和账户,登录上去查询了10W条记录,需要把这些记录拉到本地做分析 方法1,远程连接到DB服务器执行OUTFILE命令,文件存储在DB机器上,只有mysql账户的情况下 ...

  2. servlet简单用法和配置示例及说明

    学习原因和目的:   我如今所接触的项目都是bs模式的web应用,而里边基本上都是用的spring MVC和前台交互,servlet貌似用的很少.   但是即便是用spring和spring MVC, ...

  3. javascript bind

    bind的作用和apply,call类似都是改变函数的execute context,也就是runtime时this关键字的指向.但是使用方法略有不同.一个函数进行bind后可稍后执行. 定义: fu ...

  4. ORA-12519, ORA-00020异常产生原因及解决方案

    近期在做项目的过程中,使用oracle时碰到了如下两个异常: ORA-12519, TNS:no appropriate service handler found: ORA-00020:maximu ...

  5. go 版本 gRPC 环境搭建(3.0正式版)

    之前装过 gRPC 的各个测试版本,有些残余的文件,正式版的安装和之前残留的清除整理如下:   安装 go 版本的 gRPC go 的安装略过.需要 go 1.5 以上版本. $ go version ...

  6. C++ 实现Range类,用于常规遍历

    PYTHON的Range类非常好用,所以用C++来简单实现下:  // 实现Range类,用于遍历 // #include <string> class Range { public: / ...

  7. 现代工程仿真CAE技术介绍

    随着现代科学技术的发展,人们正在不断建造更为快速的交通工具.更大规模的建筑物.更大跨度的桥梁.更大功率的发电机组和更为精密的机械设备.这一切都要求工程师在设计阶段就能精确地预测出产品和工程的技术性能, ...

  8. IIS启用兼容模式设置(win2k3—Win7)

    点击添加按钮(上图),弹出下面的对话框(下图).在自定义HTTP头名处输入: X-UA-compatible 在自定义HTTP头值处输入: IE=EmulateIE7 (输入时注意不要留下空格)输入完 ...

  9. 2016-08-16: 检测函数是否存在的C++模板

    #include <iostream> struct Hello { ; } }; struct Generic {}; // SFINAE test template <typen ...

  10. C#中如何定义全局变量及在各窗体中使用全局变量

    using System; using System.Collections.Generic; using System.Drawing; using System.Windows.Forms; us ...