首先调用fork()使子进程脱离终端,父进程退出;接着在子进程中调用setsid()创建新会话并脱离控制终端;然后再次fork()防止重新获取终端;之后将工作目录改为根目录,设置umask为0;最后关闭标准输入、输出、错误流并重定向到/dev/null,进入主循环运行服务。

在Linux系统中,C++可以通过调用系统API来创建一个守护进程(Daemon Process)。守护进程是长期运行在后台的服务程序,不依赖终端,独立于用户会话。以下是创建守护进程的标准步骤和代码示例。
1. 调用fork()并让父进程退出
这是为了使子进程成为后台进程。通过fork()创建子进程后,父进程立即退出,子进程由init(或systemd)接管,脱离控制终端。
说明:这一步确保进程不是进程组的组长,为后续调用setsid()做准备。
2. 创建新会话(调用setsid)
子进程中调用setsid(),作用包括:
立即学习“C++免费学习笔记(深入)”;
- 创建一个新的会话,当前进程成为会话首进程
- 脱离控制终端(TTY)
- 成为新进程组的组长
注意:setsid()调用要求当前进程不能是进程组的组长,所以必须在第一次fork()后的子进程中调用。
3. 再次fork()防止重新获取终端
第二次fork()可确保守护进程不能重新打开控制终端。即使它调用open()访问终端设备,也不会获得控制权。
第二次子进程继续执行服务逻辑,父进程退出。
4. 改变工作目录、设置文件掩码
建议将当前工作目录改为根目录/,避免因挂载点被卸载导致问题。
同时设置文件创建掩码(umask),避免权限问题:
- chdir("/"); // 切换到根目录
- umask(0); // 清除文件模式屏蔽字
5. 关闭不需要的文件描述符
守护进程不使用标准输入、输出和错误,应关闭它们以释放资源:
- close(STDIN_FILENO);
- close(STDOUT_FILENO);
- close(STDERR_FILENO);
也可重定向到/dev/null,防止后续操作出错。
C++ 示例代码
以下是一个简单的C++守护进程创建示例:
#include#include #include #include #include #include int main() { // 第一次fork pid_t pid = fork(); if (pid < 0) { exit(1); } if (pid > 0) { exit(0); // 父进程退出 } // 子进程创建新会话 if (setsid() < 0) { exit(1); } // 第二次fork pid = fork(); if (pid < 0) { exit(1); } if (pid > 0) { exit(0); } // 修改工作目录 chdir("/"); // 设置umask umask(0); // 关闭标准流 close(STDIN_FILENO); close(STDOUT_FILENO); close(STDERR_FILENO); // 重定向标准流到 /dev/null open("/dev/null", O_RDONLY); // stdin open("/dev/null", O_WRONLY); // stdout open("/dev/null", O_WRONLY); // stderr // 守护进程主循环 while (true) { // 这里写你的服务逻辑 sleep(10); } return 0; }
编译与运行
使用g++编译:
g++ daemon.cpp -o mydaemon
运行后进程将在后台持续运行,可通过ps aux | grep mydaemon查看。
基本上就这些。只要按步骤实现,就能正确创建一个Linux守护进程。











