2.30 守护进程(1) 2.31 守护进程(2)

2.30 守护进程(1)

终端

echo $$//查看当前终端的的pid
tty//查看当前终端设备

控制终端可以操作某一个进程。
在这里插入图片描述

进程组

在这里插入图片描述

会话

在这里插入图片描述

进程组、会话、控制终端之间的关系

在这里插入图片描述
find/2 查看2重定向到dev/null设备上,|管道(创建子进程),wc -l统计 &在后台运行,为一个新的进程组
PGID为进程组ID,SID为会话ID
执行一个命令,默认是在前台执行,除非加上&号。同一时刻,只能有一个前台进程组,只有前台进程组才能享有控制终端的操作权力。

在这里插入图片描述

进程组、会话操作函数

在这里插入图片描述

守护进程

在这里插入图片描述

2.31 守护进程(2)

守护进程的创建步骤

在这里插入图片描述
若当前进程创建会话,当前进程产生进程组,进程组的组长不能和会话组长为同一进程。创建会话的进程不能为进程的组长,不能为进程的首进程
父进程死后会有shell提示符,如下:
在这里插入图片描述
(1)执行一个fork(),之后父进程退出,子进程继续后台执行的原因:防止父进程在被杀死时产生shell提示符。且确保子进程不会成为进程组的首进程。
(2)子进程调用setsid()开启一个新会话,原因:
id 100的进程调用setid()去开启一个会话,会执行下面两个步骤:
100的进程放在一个进程组中,进程组id为100,100也为会话的id。原先进程100所在的会话id为80,现在新创建出一个会话。新创建出来的会话只要和控制终端没有建立连接,那么就没有控制终端。目的:脱离控制终端(为什么调用setsid()?)。
为什么在子进程中调用?
若父进程id为100,进程组id为100。id100创建一个新会话,新会话中有进程组id100,进程id100,与原先会话冲突。若使用子进程id101创建新会话,则新会话中的进程组id为101,进程id为101,不会产生冲突

守护进程是后台(不受控制终端控制)服务(长期)进程。

写一个守护进程,每隔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);
    //将time_t类型时间转换为本地时间
    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);
	//脱离控制终端,这种方式不输出,且使用while循环,不让设置完定时器后程序即结束
    // printf("%s\n", buf);

	//asctime()系统调用
    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.更改工作目录为跟目录:
    //守护进程会在系统启动的时候被创建并一直运行直至系统被关闭。
    //如工作目录为U盘,那么U盘一直不能卸载系统文件,而跟目录一直不会卸载工作目录
    
    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;
}

显示结果
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值