一.基础概念
1.守护进程:精灵进程,在后台为用户提高服务,是一个生存周期长,通常独立于控制终端并且周期性的执行任务火处理事件发生
2.ps axj:查看守护进程
3.进程组:多个进程的集合,由于管理进程
4.组长进程:由于标识每个进程组,进程组的ID是组长进程的ID
5.getsid():获取会话ID
6.getpgrp():获取组长进程的ID
7.tail -f 文件名 :查看日志文件的写入变化
二.编程流程
1.调用fork(),产生子进程A,退出父进程
2.脱离控制终端,建立新的会话
进程与控制终端,登录会话和进程组之间的关系:进程属于一个进程组,进程组号(GID)就是进程组长的进程号(PID)。登录会话可以包含多个进程组。这些进程组共享一个控制终端。这个控制终端通常是创建进程的 shell 登录终端。 控制终端、登录会话和进程组通常是从父进程继承下来的。 我们的目的就是要摆脱它们 ,使之不受它们的影响。因此需要调用 setsid() 使子进程成为新的会话组长
setsid() 调用成功后,进程成为新的会话组长和新的进程组长,并与原来的登录会话和进程组脱离。由于会话过程对控制终端的独占性,进程同时与控制终端脱离。
3.fork()去掉子进程A的组长进程身份,禁止子进程A打开控制终端
现在,子进程A已经成为无终端的会话组长, 但它可以重新申请打开一个控制终端。可以通过使进程不再成为会话组长来禁止进程重新打开控制终端,采用的方法是再次创建一个子进程B,
4.修改工作路径为根目录
5.清空掩码
进程从创建它的父进程那里继承了文件创建掩模。它可能修改守护进程所创建的文件的存取权限。为防止这一点,将文件创建掩模清除
6.关闭文件描述符
代码:
#include<stdio.h> // 包含标准输入输出库
#include<stdlib.h> // 包含标准库函数,如exit
#include<unistd.h> // 包含Unix标准函数,如fork, setsid, sleep
#include<string.h> // 包含字符串操作函数,如memset
#include<sys/stat.h> // 包含文件状态相关函数,如umask
#include<fcntl.h> // 包含文件控制选项,如open, close
#include<time.h> // 包含时间函数,如time, asctime, localtime
int main()
{
pid_t pid = fork(); // 创建一个新的进程
if(pid != 0) // 如果当前进程不是子进程(即父进程),则退出
{
exit(0); // 退出父进程
}
setsid(); // 创建一个新的会话,该会话与终端脱离关联
pid = fork(); // 再次创建一个新的进程
if(pid != 0) // 如果当前进程不是子进程(即上一个子进程的父进程),则退出
{
exit(0); // 退出上一个子进程的父进程
}
chdir("/"); // 将当前工作目录更改为根目录"/"
umask(0); // 设置文件创建时的默认权限掩码为0,即不限制权限
// 关闭所有打开的文件描述符
for(int i=0; i<getdtablesize(); i++)
{
close(i); // 关闭文件描述符i
}
while(1) // 无限循环
{
FILE* fp = fopen("/tmp/c2305d.log","a"); // 打开或创建日志文件以追加模式
if(fp == NULL) // 如果文件打开失败
{
break; // 退出循环
}
time_t tv; // 定义时间变量
time(&tv); // 获取当前时间
fprintf(fp,"time=%s",asctime(localtime(&tv))); // 将格式化的时间写入文件
fclose(fp); // 关闭文件
sleep(3); // 休眠3秒
}
}
tail -f c2305d.log查看: