当前位置: 首页 > news >正文

Linux高阶——1027—守护进程

1、守护进程的基本流程

1、父进程创建子进程,父进程退出

守护进程是孤儿进程,但是是工程师人为创建的孤儿进程,低开销模式运行,对系统没有压力

2、子进程(守护进程)脱离控制终端,创建新会话

3、将守护进程中无用的描述符关闭(STDOUT|STDIN)

将STDERROR重定向到文件中,避免屏幕抛出异常信息,避免在使用前台时,后台的守护进程输出到屏幕上

需要把标准出错也要关闭,STDERR_FILENO,标准出错指的是,当使用open函数打开一个不存在文件时,会在终端上显示出来的错误

重定向函数dup和dup2

错误信息会通过STDERR_FILENO将异常抛出到显示器上,因此需要将错误信息重定向,不使用STDERR_FILENO流,而使用输出错误文件的文件描述符fd流

4、将守护进程的工作路径变为根目录

进程默认的工作目录一般为执行程序目录(app)

将默认工作目录(usb)改为目标电脑的根目录,避免工作目录丢失

5、修改进程umask文件权限掩码,变为0002

一般创建文件,文件权限是0664,6表示所有者权限,6表示同组用户权限,4表示其他人权限

一般电脑上初始权限mod为0666,但要与mask掩码0002做取反求与运算,得到0664

开发机的mod初始权限是0666,电脑权限掩码为0002,但目标主机的默认掩码不一定等于0002

将进程掩码改为0002,当进程创建文件时,创建出的文件掩码一定为0002

6、执行守护进程任务(间隔执行|条件触发|定时触发)

7、守护进程退出处理

测试任务

获取时间函数

用户传入数组地址,此函数将字符串时间写入数组中

设定数组中每个元素为time_t类型的变量tp

使用bzero函数初始化数组

使用time函数获取时间种子,赋值给tp

将时间种子tp传入ctime_r函数,计算出当前时间

ctime_r函数最后的返回值带'\n',因此使用tm[strlen(tm)-1]='\0'将'\n'去除

int Get_time(char* tm) 
{time_t tp; bzero(tm,1024);tp=time(NULL);ctime_r(&tp,tm);tm[strlen(tm)-1]='\0';return 0;
}

创建进程的工作

void daemon_job(void)
{int fd;if((fd=open("system.log",O_RDWR|O_CREAT,0664))==-1){perror("open failed");}char tm[1024];char log_info[4096];while(1){Get_time(tm);sprintf(log_info,"<%s>WARING war informatcion...\n",tm);write(fd,log_info,strlen(log_info));bzero(log_info,sizeof(log_info));bzero(tm,sizeof(tm));sleep(3);}
}

 创建守护进程

int create_daemon(void)
{pid_t pid;int efd;efd=open("ERROR_MSG",O_RDWR|O_CREAT,0664);pid=fork();if(pid>0){exit(0);}else if(pid==0){setsid();close(STDIN_FILENO);close(STDOUT_FILENO);dup2(efd,STDERR_FILENO);chdir("./");umask(0002);daemon_job();}else{perror("fork call failed");}return 0;
}

 总代码

#include<stdio.h>
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<errno.h>
#include<signal.h>
#include<pthread.h>
#include<sys/fcntl.h>
#include<string.h>int Get_time(char* tm) 
{time_t tp; bzero(tm,1024);tp=time(NULL);ctime_r(&tp,tm);tm[strlen(tm)-1]='\0';return 0;
}void daemon_job(void)
{int fd;if((fd=open("system.log",O_RDWR|O_CREAT,0664))==-1){perror("open failed");}char tm[1024];char log_info[4096];while(1){Get_time(tm);sprintf(log_info,"<%s>WARING war informatcion...\n",tm);write(fd,log_info,strlen(log_info));bzero(log_info,sizeof(log_info));bzero(tm,sizeof(tm));sleep(3);}
}int create_daemon(void)
{pid_t pid;int efd;efd=open("ERROR_MSG",O_RDWR|O_CREAT,0664);pid=fork();if(pid>0){exit(0);}else if(pid==0){setsid();close(STDIN_FILENO);close(STDOUT_FILENO);dup2(efd,STDERR_FILENO);chdir("./");umask(0002);daemon_job();}else{perror("fork call failed");}return 0;
}int main()
{create_daemon();return 0;
}

创建一个system.log的日志文件,每隔3秒,向日志文件中写入当前系统时间

eg:<系统时间>WARNING,test war message

2、实现Linux操作系统开机启动

shell脚本

脚本池里有许多脚本,每个脚本可以启动一个程序

虚拟机操作系统LINUX OS启动时,会将脚本池里的所有shell脚本对应的程序启动

编写一个shell脚本,让此脚本可以定位并启动守护程序

步骤:

1、将脚本放进系统脚本文件夹/etc/init.d/

2、将自定义脚本通过update-rc.d加入系统脚本池

3、!/bin/bash声明脚本中使用的shell版本

sudo mv start_daemon /etc/init.d/

cd /etc/init.d

sudo update-rc.d shell_name start 99 2.——添加启动

sudo update-rc.d shell_name remove——删除启动

shell脚本——命令的容器,可以将大量命令写入脚本,最后进行批量执行和处理,有了脚本后使用命令更方便


http://www.mrgr.cn/news/63995.html

相关文章:

  • flink cdc oceanbase(binlog模式)
  • QT给端口扫描工程增加线程2
  • 【HarmonyOS-ArkTS语言】面向对象【合集】
  • Spring源码分析之事件机制——观察者模式(二)
  • 嵌入式岗位面试八股文(篇三 操作系统(上))
  • 大模型原理解释
  • LeetCode207. 课程表(2024秋季每日一题 55)
  • Mybatis-plus入门教程
  • 【深度学习基础】常用图像卷积核类型
  • 基于STM32的智能水族箱控制系统设计
  • 大学城水电资源管理:Spring Boot解决方案
  • 俗人只知《老子》该书,却不知李耳其人,可悲
  • 大屏可视化管理系统建设方案书(word原件)
  • 六 在WEB中应用MyBatis(使用MVC架构模式)
  • 这篇文章,教你如何看清流量卡套路!
  • 服务端监控工具:Nmon使用方法
  • 【WebApi】C# webapi 后端接收部分属性
  • python 五子棋小游戏
  • 《深入浅出HTTPS​​​》读书笔记(3):HTTP本身的安全问题
  • Maven(20) 如何使用Maven进行版本管理?
  • Weblogic漏洞复现(Vulhub)
  • 项目实战:基于Linux的Flappy bird游戏开发
  • Mac下载 安装MIMIC-IV 3.0数据集
  • 数据分析SPSS面试题及参考答案
  • 机器学习在运维中的应用
  • 多个JDK版本之间的切换