Linux 系统编程:打造“不死之身”——手把手教你编写守护进程 (Daemon)

资料合集(福利需要仔细找)
链接: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),子进程继续。
  • 目的
    1. 让进程在后台运行(Shell 以为父进程结束了,就会返回命令提示符)。
    2. 关键点:保证子进程不是进程组组长。这是下一步调用 setsid() 的前提条件。

第二步:创建新会话 (setsid)

  • 动作:子进程调用 setsid() 函数。
  • 目的:这是最重要的一步!
    1. 子进程成为新会话的首进程(Session Leader)。
    2. 子进程成为新进程组的组长。
    3. 彻底脱离原控制终端。从此,该进程真正获得了“独立”。

第三步:改变当前工作目录 (chdir)

  • 动作chdir("/")
  • 目的:防止占用可卸载的文件系统。如果守护进程在 /mnt/usb 下启动,它会一直占用该目录,导致 USB 无法卸载。通常改到根目录 //tmp

第四步:重设文件权限掩码 (umask)

  • 动作umask(0)
  • 目的:继承来的文件创建掩码可能会屏蔽掉某些权限(如禁止写)。设置为 0 可以增加守护进程的灵活性,确保它创建的文件具有请求的所有权限。

第五步:关闭/重定向文件描述符

  • 动作:关闭标准输入(0)、标准输出(1)、标准错误(2)。
  • 目的:既然脱离了终端,printf 打印给谁看?保留这些描述符不仅浪费资源,还可能导致安全隐患。
  • 最佳实践:通常将它们重定向到 /dev/null(黑洞设备)。

四、 代码实战:编写一个记录时间的守护进程

我们要编写一个程序,它启动后会变成守护进程,每隔 2 秒向日志文件中写入当前时间。

1. 代码实现 (my_daemon.c)

#include <stdio.h>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

web安全工具库

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值