守护进程
- 何谓守护进程
- 常见守护进程
- 进程查看命令ps
- kill命令
- 编写简单守护进程
- 守护进程的父进程
何谓守护进程
-
daemon,表示守护进程,简称为d(进程名后面带d的基本就是守护进程)
-
长期运行(一般是开机运行直到关机时关闭)
-
与控制台脱离(普通进程都和运行该进程的控制台相绑定,表现为如果终端被强制关闭了则这个终端中运行的所有进程都被会关闭,背后的问题还在于会话)
-
服务器(Server),服务器程序就是一个一直在运行的程序,可以给我们提供某种服务(譬如nfs服务器给我们提供nfs通信方式),当我们程序需要这种服务时我们可以调用服务器程序(和服务器程序通信以得到服务器程序的帮助)来进程这种服务操作。服务器程序一般都实现为守护进程。
常见守护进程
- syslogd,系统日志守护进程,提供syslog功能。
- cron,cron进程用来实现操作系统的时间管理,linux中实现定时执行程序的功能就要用到cron。
进程查看命令ps
ps -ajx //偏向显示各种有关的ID号
ps -aux //偏向显示进程各种占用资源
ps -aux | grep syslogd // grep 过滤提取想看的进程名"syslogd"
kill命令
kill -信号编号 进程ID //向一个进程发送一个信号
kill -9 xxx //将向xxx这个进程发送9号信号,也就是要结束进程
编写简单守护进程
-
子进程等待父进程退出
-
子进程使用setsid创建新的会话期,脱离控制台
-
调用chdir将当前工作目录设置为/
-
umask设置为0以取消任何文件权限屏蔽
-
调用sysconf函数关闭所有文件描述符
-
调用open函数,将0、1、2定位到/dev/null,赋值可读可写权限
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
// 函数作用就是把调用该函数的进程变成一个守护进程
void creat_daemon()
{
pid_t pid = 0;
pid = fork();
if(pid == -1)
{
perror("fork");
exit(-1);
}
// 父进程直接退出
if(pid > 0)
{
exit(0);
}
// 执行到这里就是子进程
// setsid将当前进程设置为一个新的会话期session,目的就是让当前进程
// 脱离控制台。
pid = setsid();
if(pid < 0)
{
perror("setsid");
exit(-1);
}
// 将当前进程工作目录设置为根目录
chdir("/");
// umask设置为0确保将来进程有最大的文件操作权限
umask(0);
int i = 0;
// 关闭所有文件描述符
// 先要获取当前系统中所允许打开的最大文件描述符数目
for(i = 0; i < sysconf(_SC_OPEN_MAX);i++)
{
close(i);
}
open("/dev/null", O_RDWR);
open("/dev/null", O_RDWR);
open("/dev/null", O_RDWR);
}
int main()
{
creat_daemon();
while(1)
{
printf("This is my daemon!\n");
}
return 0;
}
运行./9_daemon
再通过
ps -ajx
查看我自己编写的守护进程“9_daemon”
如果想杀死守护进程,则输入
kill -9 pid
守护进程的父进程
在 Linux 系统中,守护进程的父进程通常是初始化进程。在现代 Linux 发行版中,这通常是 systemd 或者类似的进程管理器。systemd 是一个系统和服务管理器,负责启动系统服务和守护进程,并管理它们的生命周期。
因此,如果你在 Linux 系统中启动一个守护进程,其父进程很可能是 systemd 进程或者由 systemd --user 管理的用户级进程。这是因为 systemd 是 Linux 中常见的进程管理器,负责在系统启动时启动和管理各种服务和守护进程。
顺着ppid找到 9.daemon 守护进程的父进程号为1493,该进程恰好为systemd --user进程