【计算机网络】日志与守护进程

文章目录

  • 日志
    • 日志的创建
      • logmessage 函数
        • 日志左边部分实现
        • 日志右边部分实现
    • 完整代码
      • log.hpp(整体实现)
      • err.hpp (错误信息枚举)
  • 守护进程
    • PGID SID TTY 的介绍
    • shell中控制进程组的方式
      • 结论
    • 为什么要有守护进程存在?
    • 守护进程的创建
      • 使用守护进程的条件
      • 守护进程化的函数
      • 自己实现守护进程化
        • 解决组长问题
        • 忽略信号
        • 处理 0 1 2 问题
        • 退出守护进程
    • 完整代码
      • err.hpp(错误信息枚举)
      • daemon.hpp(整体实现)

日志

一般使用cout进行打印,但是cout打印是不规范的
实际上 是采用日志进行打印的

日志的创建

创建一个 log.hpp


日志有自己的日志等级

通过枚举,分别为 调试 常规 告警 一般错误 致命错误 未知错误


logmessage 函数

定义一个函数 logmessage,参数level 为日志等级 ,
为了按照可变参数的方式,来进行格式化输出,所以设置一个format 以及…可变参数(可以给c函数传递任意个数的参数)

日志左边部分实现

输入 man snprintf

将可变参数的内容显示到str字符串中


获取日志等级

设置一个字符串 level_string ,通过tolevelstring函数 将数字转化为字符串


获取时间

输入 man localtime

将time_t转换为 struct tm 结构体类型


该结构体包含 秒 分 时 天


输入 man 3 time


在这里插入图片描述
通过gettime函数 获取时间


日志右边部分实现

为了处理可变参数部分,所以使用vsprintf
输入 man snprintf


将写好的数据放到logRight中

完整代码

log.hpp(整体实现)

#pragma once 
#include<iostream>
#include<string.h>
#include<cstdio>
#include<cstring>
#include<cstdarg>
#include<unistd.h>
#include<sys/types.h>
#include<time.h>const std::string  filename="tecpserver.log";//日志等级
enum{DEBUG=0, // 用于调试INFO  ,  //1 常规WARNING, //2 告警ERROR ,  //3  一般错误FATAL ,  //4 致命错误UKNOWN//未知错误
};static  std::string tolevelstring(int level)//将数字转化为字符串
{switch(level){case DEBUG : return "DEBUG";case INFO  : return "INFO";case WARNING : return "WARNING";case  ERROR : return "ERROR";case FATAL : return "TATAL";default: return "UKNOWN";}
}
std::string gettime()//获取时间
{time_t curr=time(nullptr);//获取time_tstruct tm *tmp=localtime(&curr);//将time_t 转换为 struct tm结构体char buffer[128];snprintf(buffer,sizeof(buffer),"%d-%d-%d %d:%d:%d",tmp->tm_year+1900,tmp->tm_mon+1,tmp->tm_mday,tmp->tm_hour,tmp->tm_min,tmp->tm_sec);return buffer;}
void logmessage(int level, const char*format,...)
{//日志左边部分的实现char logLeft[1024];std::string level_string=tolevelstring(level);std::string curr_time=gettime();snprintf(logLeft,sizeof(logLeft),"%s %s %d",level_string.c_str(),curr_time.c_str());//日志右边部分的实现char logRight[1024]; va_list p;//p可以看作是1字节的指针va_start(p,format);//将p指向最开始vsnprintf(logRight,sizeof(logRight),format,p);va_end(p);//将指针置空//打印日志 printf("%s%s\n",logLeft,logRight);//保存到文件中FILE*fp=fopen( filename.c_str(),"a");//以追加的方式 将filename文件打开//fopen打开失败 返回空指针if(fp==nullptr){return;}fprintf(fp,"%s%s\n",logLeft,logRight);//将对应的信息格式化到流中fflush(fp);//刷新缓冲区fclose(fp);
}

err.hpp (错误信息枚举)

#pragma once enum
{USAGE_ERR=1,SOCKET_ERR,//2BIND_ERR,//3LISTEN_ERR,//4SETSID_ERR,//5OPEN_ERR//6
};

守护进程

网络服务一定在任何时候都能访问,所以这个服务不能受任何用户的登录或者注销各种行为的影响
所以需要将进程进行守护进程化

PGID SID TTY 的介绍

在后台运行sleep 10000


PPID是bash的PID值
PGID是 进程组 (PGID相同就为同一个进程组,以从第一个进程进行命名)
SID 是 会话ID
TTY是 终端 若为?,则说明跟终端没有关系,若为具体的如pts/5,则为终端文件


在终端2中输入,在终端1中可以查看到 两者的PGID相同,所以属于同一个进程组,并且以sleep 1000 作为组长


通过查询会话ID 21668,发现bash的PID PGUD SID 都为21668

shell中控制进程组的方式

查询后台任务 jobs


当再次输入sleep 5000 进行后台运行时,发现前面的编号变成2
该编号为 任务编号


将某一任务提到前台运行 fg + 任务编号

当把1号任务提到前台后,再次使用jobs查询后台任务,就查不到1号任务了
并且其他任务并不受影响


把2号任务提到前台,使用 ctrl z 让服务暂停起来
在暂停后,任务会自动切换到后台


输入 bg 2,让2号任务在后台跑起来

结论

1. 进程组分为 前台任务 和 后台任务

在终端2中创建后台任务和前台任务,在终端1中查询发现,后台任务的(PGID)进程组 和 (SID)会话ID相同 ,而与后台的不同


2. 如果后台任务提到前台,老的前天任务就无法运行

将任务编号为1的后台任务 使用 fg 提到前台后 ,输入 ls pwd 等 指令是没有作用的
会话中 ,只能有一个前台任务在运行
所以当 使用 ctrl c 将1号任务退出后,bash把自己变成了前台任务,所以又可以运行了

为什么要有守护进程存在?

若登录就是创建一个会话,启动进程,会话内部有bash任务,在当前会话中创建新的前后台任务,那如果退出呢?

当退出时,就会销毁会话可能会影响会话内部的所有任务

网络服务器为了不受到用户登录注销的影响,网络服务器 通常以守护进程的方式运行

守护进程的创建

输入 man 2 setsid

设置一个会话,以进程组的组长ID作为新的会话ID

若返回成功,则返回调用进程的PID,若返回失败,则返回-1并设置错误码


想要调用setsid,不可以是组长

如:在一家公司中你是组长,有一天你想不干了 出去创业 是不可以的,因为你手底下有一堆组员
所以要成功出去创业,就必须卸任你的组长身份

使用守护进程的条件

1.忽略异常
2.对 0(标准输入) 1(标准输出) 2(标准错误) 作特殊处理
3.进程的工作路径 可能要更改
4.守护进程是一个全局的进程,不想在某一个用户的目录下,所以从整个系统中从最开始进行索引某些文件

守护进程化的函数

输入 man daemon,提供守护进程化的函数

第一个参数表示 是否更改 工作目录,默认不要改,改为1表示为真
第二个参数表示 要不要关闭 0 1 2, 默认不关

大部分情况下,都是自己实现守护进程,而不是调用该函数

自己实现守护进程化

解决组长问题

当启动时,是在bash中新起一个任务,只有一个进程自成进程组,所以自成组长,操作不被允许

成为组长的一般都是组中的第一个进程,所以只需使其不为第一个进程即可


输入 man fork,创建子进程

fork的返回值:父进程返回子进程的PID值,子进程返回0,失败返回-1


当fork>0时,说明为父进程,则让父进程退出,只剩下子进程,子进程不是进程的第一个,也就不是组长,就可以成功调用setsid

忽略信号

signal的第一个参数 表示 信号 ,第二个参数表示对指定动作的信号设定自定义处理动作


SIGPIPE 表示13号信号


SIG_IGN 为 自定义处理信号处理函数

把1强制转化成函数指针类型 即忽略信号

对13号信号 进行忽略


SIGCHLD信号
子进程在运行时会退出,若父进程不关心子进程退出,子进程就会变成僵尸状态
父进程要使用 wait/waitpid去等待子进程 回收僵尸,获取子进程的退出结果
即父进程进行阻塞式等待(什么都不干,就等待子进程的退出结果)
子进程要退出时,会向父进程发信号 SIGCHLD

所以同样对 SIGCHLD信号 进行忽略


处理 0 1 2 问题

使用日志打印,所以导致有很多输出结果,但输出结果不想往显示器上面打印,所以就需要处理标准输入 标准输出 标准错误


Linux系统提供一个 dev null的字符设备

向dev null 中写入,都会被丢弃 ,从这个文件读什么都读不到 ,立马直接返回


输入 man 2 open,打开文件

若返回成功,则返回 文件描述符,若返回失败,则返回 -1 并将错误码返回
O_RDWR : 读写的方式


重定向函数 :输入 man dup2

可以直接将文件打开,使用dup2重定向
输出重定向对应的文件描述符是1
假设其文件描述符是fd
newfd为oldfd的一份拷贝,最后只剩下oldfd
dup2(fd,1) 即 将标准输出流 重定向到 文件描述符fd中


退出守护进程

输入 kill -9 + 守护进程的PID,即可退出守护进程

完整代码

err.hpp(错误信息枚举)

#pragma once enum
{USAGE_ERR=1,SOCKET_ERR,//2BIND_ERR,//3LISTEN_ERR,//4SETSID_ERR,//5OPEN_ERR//6
};

daemon.hpp(整体实现)

#pragma once
#include<unistd.h>
#include<signal.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include"log.hpp"
#include"err.hpp"void Daemon()//自己实现服务器的守护进程化{//1.忽略信号signal(SIGPIPE,SIG_IGN);//忽略信号signal(SIGCHLD,SIG_IGN);//2.不要成为组长    if(fork()>0)//说明为父进程,则让父进程直接退出{exit(0);}//只剩下子进程//3.新建会话,自己成为会话的话首进程pid_t ret=setsid();if((int)ret==-1)//守护进程失败{logmessage(FATAL,"deamon error,code:%d,string :%s",errno,strerror(errno));exit(SETSID_ERR);//终止程序}//4.可以更改守护进程的工作路径//5.处理 0 1 2 问题int fd=open("/dev/null",O_RDWR);//以读写的方式打开字符设备if(fd<0){logmessage(FATAL,"deamon error,code:%d,string :%s",errno,strerror(errno));exit(OPEN_ERR);//终止程序}   //将标准输入 输出错误 重定向到字符设备中dup2(fd,0);dup2(fd,1);dup2(fd,2);close(fd);} 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/101336.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2023国赛数学建模A题B题C题D题资料思路汇总 高教社杯

本次比赛我们将会全程更新思路模型及代码&#xff0c;大家查看文末名片获取 之前国赛相关的资料和助攻可以查看 2022数学建模国赛C题思路分析_2022年数学建模c题思路_UST数模社_的博客-CSDN博客 2022国赛数学建模A题B题C题D题资料思路汇总 高教社杯_2022国赛a题题目_UST数模…

前端通信(渲染、http、缓存、异步、跨域)自用笔记

SSR/CSR&#xff1a;HTML拼接&#xff1f;网页源码&#xff1f;SEO/交互性 SSR &#xff08;server side render&#xff09;服务端渲染&#xff0c;是指由服务侧&#xff08;server side&#xff09;完成页面的DOM结构拼接&#xff0c;然后发送到浏览器&#xff0c;为其绑定状…

浅谈泛在电力物联网发展形态与技术挑战

安科瑞 华楠 摘 要&#xff1a;泛在电力物联网是当前智能电网发展的一个方向。首先&#xff0c;总结了泛在电力物联网的主要作用和价值体现&#xff1b;其次&#xff0c;从智能电网各个环节概述了物联网技术在电力领域的已有研究和应用基础&#xff1b;进而&#xff0c;构思并…

手机无人直播软件,有哪些优势?

近年来&#xff0c;随着手机直播的流行和直播带货的市场越来越大&#xff0c;手机无人直播软件成为许多商家开播带货的首选。在这个领域里&#xff0c;声音人无人直播系统以其独特的优势&#xff0c;成为市场上备受瞩目的产品。接下来&#xff0c;我们将探讨手机无人直播软件给…

基于jeecg-boot的flowable流程加签功能实现

更多nbcio-boot功能请看演示系统 gitee源代码地址 后端代码&#xff1a; https://gitee.com/nbacheng/nbcio-boot 前端代码&#xff1a;https://gitee.com/nbacheng/nbcio-vue.git 在线演示&#xff08;包括H5&#xff09; &#xff1a; http://122.227.135.243:9888 今天我…

uni-app打包后安卓不显示地图及相关操作详解

新公司最近用uni-app写app&#xff0c;之前的代码有很多问题&#xff0c;正好趁着改bug的时间学习下uni-app。 问题现象&#xff1a; 使用uni-app在浏览器调试的时候&#xff0c;地图是展示的&#xff0c;但是打包完成后&#xff0c;在app端是空白的。咱第一次写app&#xff…

视频云存储/安防监控EasyCVR视频汇聚平台分发rtsp流时,出现“用户已过期”提示该如何解决?

视频云存储/安防监控EasyCVR视频汇聚平台基于云边端智能协同&#xff0c;支持海量视频的轻量化接入与汇聚、转码与处理、全网智能分发、视频集中存储等。音视频流媒体视频平台EasyCVR拓展性强&#xff0c;视频能力丰富&#xff0c;具体可实现视频监控直播、视频轮播、视频录像、…

2023企业Hyper-V备份解决方案!

虚拟技术已成为IT基础设施的核心元素&#xff0c;企业长期依赖虚拟机&#xff08;VM&#xff09;为应用程序提供工作负载。Hyper-V是一款广受欢迎的虚拟化平台&#xff0c;多年来已日臻完善&#xff0c;并在各类规模的公司中得到广泛应用。 随着虚拟机使用的普及&…

FPGA使用MIG调用SODIMM内存条接口教程,提供vivado工程源码和技术支持

目录 1、前言免责声明 2、SODIMM内存条简介3、设计思路框架视频输入视频缓存MIG配置调用SODIMM内存条VGA时序视频输出 4、vivado工程详解5、上板调试验证6、福利&#xff1a;工程代码的获取 1、前言 FPGA应用中&#xff0c;数据缓存是一大重点&#xff0c;不管是图像处理还是A…

搭建开发环境-操作系统篇(一键搭建开发环境)

概述 所谓工欲善其事必先利其器&#xff0c;搭环境往往是开发过程中卡出很多初学者的拦路虎。 对于很多老鸟来说&#xff0c;很多东西都已经习惯成自然&#xff0c;也就没有刻意和初学者说。但对于很多初学者&#xff0c;却是受益良多。 这个系列&#xff0c;先从操作系统开始…

ES基础操作

1.创建索引 在 Postman 中&#xff0c;向 ES 服务器发 PUT 请求 &#xff1a; http://127.0.0.1:9200/shopping 后台日志 重复发送 PUT 请求添加索引 &#xff1a; http://127.0.0.1:9200/shopping &#xff0c;会返回错误信息 : 2.获取单个索引相关信息 在 Postman 中&#…

IDC发布《亚太决策支持型分析数据平台评估》报告,亚马逊云科技位列“领导者”类别

日前&#xff0c;领先的IT市场研究和咨询公司IDC发布《2023年亚太地区&#xff08;不含日本&#xff09;决策支持型分析数据平台供应商评估》1报告&#xff0c;亚马逊云科技位列“领导者”类别。IDC认为&#xff0c;亚马逊云科技在解决方案的协同性、敏捷性、完整性、及时性、经…

[C++] string类常用接口的模拟实现

文章目录 1、前言2、遍历2.1 operator[ ]下标方式2.2 迭代器2.3 范围for2.4 c_str 3、容量相关3.1 size&#xff08;大小&#xff09;3.2 capacity&#xff08;容量&#xff09;3.3 empty&#xff08;判空&#xff09;3.4 clear&#xff08;清理&#xff09;3.5 reserve3.6 res…

idea连接linux远程docker详细教程操作

1&#xff1a;修改docker配置文件docker.service vi /usr/lib/systemd/system/docker.service2&#xff1a;找到 ExecStart&#xff0c;在最后面添加 -H tcp://0.0.0.0:2375 # for containers run by docker ExecStart/usr/bin/dockerd -H fd:// --containerd/run/containerd/…

Android动态添加和删除控件/布局

一、引言 最近在研究RecyclerView二级列表的使用方法&#xff0c;需要实现的效果如下。 然后查了一些博客&#xff0c;觉得实现方式太过复杂&#xff0c;而且这种方式也不是特别受推荐&#xff0c;所以请教了别人&#xff0c;得到了一种感觉还不错的实现方式。实现的思路为&…

航空电子设备中的TSN通讯架构—直升机

前言 以太网正在迅速取代传统网络&#xff0c;成为航空电子设备和任务系统的核心高速网络。本文提出了以太网时间敏感网络(TSN)在航空电子设备上应用的技术优势问题。在实际应用中&#xff0c;TSN已成为一个具有丰富的机制和协议的工具箱&#xff0c;可满足与时间和可靠性相关…

从 0 到 1 读懂:哈希表

哈希表 一、什么是哈希表&#xff1f;二、两种散列函数构造方法1、直接定址法2、除留余数法&#xff08;常用&#xff09; 三、散列地址冲突四、常用冲突处理1、负载因子调节&#xff08;减少冲突概率&#xff09;2、开放定址法&#xff08;闭散列&#xff09;&#xff08;1&am…

【运维】linkis1.3.2添加jdbc引擎(添加mysql、greenplum、starrocks、doris数据源查询)与配合多数据源管理提交任务初探

文章目录 一. 引擎的安装1. 前置工作2. 获取引擎插件3. 上传和加载4. 引擎刷新4.1. 重启刷新4.2. 检查引擎是否刷新成功 二. 测试mysql、starrocks与doris数据库1. 通过shell提交任务2. 通过(IDE)shell进行提交3. 通过接口提交 三. 添加greenplum四. 通过linkis的数据源管理提交…

【韩顺平 零基础30天学会Java】程序流程控制(2days)

day1 程序流程控制&#xff1a;顺序控制、分支控制、循环控制 顺序控制&#xff1a;从上到下逐行地执行&#xff0c;中间没有任何判断和跳转。 Java中定义变量时要采用合法的前向引用。 分支控制if-else&#xff1a;单分支、双分支和多分支。 单分支 import java.util.Scann…

Appium-移动端自动测试框架,如何入门?

Appium是一个开源跨平台移动应用自动化测试框架。 既然只是想学习下Appium如何入门&#xff0c;那么我们就直奔主题。文章结构如下&#xff1a; 1、为什么要使用Appium&#xff1f; 2、如何搭建Appium工具环境?(超详细&#xff09; 3、通过demo演示Appium的使用 4、Appium如何…