1.终端、会话、进程组
三点注意:1.我们在终端上运行的进程都是shell进程fork出来的子进程
2.发信号是给前台进程发的
3.后台进程的设置方法是加一个& 比如: ./a.out&
注意:进程组首进程无需是最后一个离开进程组的成员
理解:当控制终端的连接建立起来之后,会话首进程会成为该终端的控制进程。
当用户登录到系统并启动shell时,会话(session)就被创建了。这个会话可以包含多个进程组(process group),每个进程组都有一个PGID(Process Group ID)。 在该会话中的第一个进程称为首进程(init process),它是由操作系统自动创建的,并且通常是init或systemd等程序。当控制终端连接建立起来之后,该终端将成为该会话的控制终端(control terminal)。 此时,如果没有其他进程已经与该终端关联,则当前运行的首进程将成为该终端的控制进程(control process),也就是说,它将拥有对于控制终端进行输入和输出操作以及接收信号的特权。
一张图了解他们之间的关系
2.守护进程的概念
文件描述符0,1,2分别对应了in,out,perror
/dev/null会将接收到的东西全部删除
守护进程一般在home目录下创建一个文件用来写日志或查看错误信息
测试代码:
/*
写一个守护进程,每隔2s获取一下系统时间,将这个时间写入到磁盘文件中。
*/
#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/time.h>
#include <signal.h>
#include <time.h>
#include <stdlib.h>
#include <string.h>
void work(int num) {
// 捕捉到信号之后,获取系统时间,写入磁盘文件
time_t tm = time(NULL);
struct tm * loc = localtime(&tm);
// char buf[1024];
// sprintf(buf, "%d-%d-%d %d:%d:%d\n",loc->tm_year,loc->tm_mon
// ,loc->tm_mday, loc->tm_hour, loc->tm_min, loc->tm_sec);
// printf("%s\n", buf);
char * str = asctime(loc);
int fd = open("time.txt", O_RDWR | O_CREAT | O_APPEND, 0664);
write(fd ,str, strlen(str));
close(fd);
}
int main() {
// 1.创建子进程,退出父进程
pid_t pid = fork();
if(pid > 0) {
exit(0);
}
// 2.将子进程重新创建一个会话
setsid();
// 3.设置掩码
umask(022);
// 4.更改工作目录
chdir("/home/nowcoder/");
// 5. 关闭、重定向文件描述符
int fd = open("/dev/null", O_RDWR);
dup2(fd, STDIN_FILENO);
dup2(fd, STDOUT_FILENO);
dup2(fd, STDERR_FILENO);
// 6.业务逻辑
// 捕捉定时信号
struct sigaction act;
act.sa_flags = 0;
act.sa_handler = work;
sigemptyset(&act.sa_mask);
sigaction(SIGALRM, &act, NULL);
struct itimerval val;
val.it_value.tv_sec = 2;
val.it_value.tv_usec = 0;
val.it_interval.tv_sec = 2;
val.it_interval.tv_usec = 0;
// 创建定时器
setitimer(ITIMER_REAL, &val, NULL);
// 不让进程结束
while(1) {
sleep(10);
}
return 0;
}