资料合集(福利需要仔细找)
链接:https://pan.quark.cn/s/b0a2f36933de
在 Linux 系统中,有这样一类“幕后英雄”。它们通常在系统启动时运行,在后台默默工作,直到系统关闭。它们不和任何终端关联,不受用户登录或注销的影响。
这类进程被称为守护进程(Daemon Process)。
你一定见过它们:mysqld(MySQL数据库服务)、httpd(Apache Web服务)、sshd(SSH远程服务)。注意到了吗?它们的名字通常以 d 结尾。今天,我们就来揭开守护进程的神秘面纱,并写一个属于自己的守护进程。
一、 什么是守护进程?
1. 核心定义
守护进程(Daemon)是运行在后台的一种特殊进程。它独立于控制终端(Control Terminal),周期性地执行某种任务或等待处理某些发生的事件。
2. 四大特征
根据课堂笔记,守护进程具备以下显著特征:
- 脱离终端:它不属于任何终端,因此你无法通过终端给它发送信号(比如 Ctrl+C 或 Ctrl+\ 对它无效)。在
ps命令中,它的TTY一列显示为?。 - 生命周期长:通常随系统启动而生,随系统关闭而死,不受普通用户登录/退出的影响。
- 后台运行:不占以前台 Shell,默默执行任务(如日志轮转、数据库请求处理)。
- 命名规范:虽然不是强制的,但习惯上守护进程的名称会以
d结尾。
二、 为什么要用守护进程?
想象一下,如果你启动了一个 Web 服务器(如 Nginx),结果当你关闭 SSH 终端时,Web 服务器也跟着关闭了,这合理吗?显然不合理。
守护进程的作用正是为了保证服务 7×24小时不间断运行。
- 服务提供:数据库、Web服务器、文件服务器。
- 系统任务:定期清理缓存、作业调度(cron)。
三、 守护进程的创建步骤(核心考点)
将一个普通进程“进化”为守护进程,需要经过一套严格的五步法流程。每一步都有其深层的系统原理。
第一步:创建子进程,父进程退出 (Fork & Exit)
- 动作:执行
fork(),父进程exit(0),子进程继续。 - 目的:
- 让进程在后台运行(Shell 以为父进程结束了,就会返回命令提示符)。
- 关键点:保证子进程不是进程组组长。这是下一步调用
setsid()的前提条件。
第二步:创建新会话 (setsid)
- 动作:子进程调用
setsid()函数。 - 目的:这是最重要的一步!
- 子进程成为新会话的首进程(Session Leader)。
- 子进程成为新进程组的组长。
- 彻底脱离原控制终端。从此,该进程真正获得了“独立”。
第三步:改变当前工作目录 (chdir)
- 动作:
chdir("/")。 - 目的:防止占用可卸载的文件系统。如果守护进程在
/mnt/usb下启动,它会一直占用该目录,导致 USB 无法卸载。通常改到根目录/或/tmp。
第四步:重设文件权限掩码 (umask)
- 动作:
umask(0)。 - 目的:继承来的文件创建掩码可能会屏蔽掉某些权限(如禁止写)。设置为 0 可以增加守护进程的灵活性,确保它创建的文件具有请求的所有权限。
第五步:关闭/重定向文件描述符
- 动作:关闭标准输入(0)、标准输出(1)、标准错误(2)。
- 目的:既然脱离了终端,
printf打印给谁看?保留这些描述符不仅浪费资源,还可能导致安全隐患。 - 最佳实践:通常将它们重定向到
/dev/null(黑洞设备)。
四、 代码实战:编写一个记录时间的守护进程
我们要编写一个程序,它启动后会变成守护进程,每隔 2 秒向日志文件中写入当前时间。
1. 代码实现 (my_daemon.c)
#include <stdio.h>

最低0.47元/天 解锁文章

被折叠的 条评论
为什么被折叠?



