守护进程
守护进程的概念
Ssh、打印、ftp等均属于守护进程。进程类型为d。
特点:始终在后台运行,独立于任何终端,周期性的执行某种任务或等待处理特定事件。
会话和控制终端:linux以会话、进程组的方式管理进程,每个进程属于一个进程组,会话是一个或多个进程的集合。通常用户打开一个终端时,系统会创建一个会话,shell为会话的首进程,所有通过该终端运行的进程都属于这个会话,会话最多只能由一个控制终端,当终端关闭时,所有相关进程会被结束,而守护进程则不受影响。
守护进程的创建步骤:
1.创建子进程,父进程退出,子进程变成孤儿进程,被init进程收养,子进程在后台运行,但依然依附于终端。
2.子进程创建新会话setid(),子进程成为新的会话组长,脱离原先的终端
3.更改当前工作目录chdir(”/”)或chdir(“/tmp”),其工作目录不能被卸载,重新设定当前工作目录cwd
4.重设文件权限掩码umask(0),只影响当前进程
5.关闭所有从父进程中打开的文件描述符,for(i = 0; i<getdtablesize();i++) close(i);其中getdtablesize()返回该进程最大文件描述符的值,子进程不知道父进程打开多少文件的情况,脱离终端,其中每个进程默认打开的stdin/stdout/stderr已无法在使用,也需清理掉
练习每隔1s向time.log写入系统时间:
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <wait.h>
#include <unistd.h>
#include <signal.h>
#include <time.h>
int main()
{
pid_t pid;
FILE *fp;
time_t t;
int i;
if((pid = fork())<0){
perror("fork");
exit(-1);
}
if(pid > 0){
exit(0);
}
else if(pid == 0){
setsid();
chdir("/tmp");
umask(0);
for(i = 0; i<getdtablesize();i++){
close(i);
}
while(1){
fp = fopen("time.log","a");
time(&t);
fprintf(fp,"%s",ctime(&t));
fflush(fp);
sleep(1);
}
}
}
结果验证:
在守护进程工作目录tmp下,用cat命令查看文件time.log。