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

文章目录

  • 日志
    • 日志的创建
      • 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.mzph.cn/news/53286.shtml

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

相关文章

算法 for GAMES

栈 #include <iostream> #include <stack>int main() {std::stack<int> intStack;// 压入元素到堆栈intStack.push(5);intStack.push(10);intStack.push(15);// 查看堆栈顶部元素std::cout << "Top element: " << intStack.top() <…

免费的png打包plist工具CppTextu,一款把若干资源图片拼接为一张大图的免费工具

经常做游戏打包贴图的都知道&#xff0c;要把图片打包为一张或多张大图&#xff0c;要使用打包工具TexturePacker。 TexturePacker官方版可以直接导入PSD、SWF、PNG、BMP等常见的图片格式&#xff0c;主要用于网页、游戏和动画的制作&#xff0c;它可以将多个小图片汇聚成一个…

kafka--技术文档-基本概念-《快速了解kafka》

学习一种新的消息中间键&#xff0c;卡夫卡&#xff01;&#xff01;&#xff01; 官网网址 Apache Kafka 基本概念 Kafka是一种开源的分布式流处理平台&#xff0c;由Apache软件基金会开发&#xff0c;用Scala和Java编写。它是一个高吞吐量的分布式发布订阅消息系统&#xf…

QCC_BES 音频重采样算法实现

+V hezkz17进数字音频系统研究开发交流答疑群(课题组) 这段代码是一个用于将音频数据进行立体声重采样的函数。以下是对代码的解读: 函数接受以下参数: pcm_buf:16位有符号整型的音频缓冲区,存储了输入的音频数据。pcm_len:音频缓冲区的长度。mic1:16位有符号整型的音频…

CSS内边距和外边距属性

外边距属性用margin&#xff1b;padding属性叫填充&#xff0c;或者也叫内边距&#xff1b; margin:标签与标签的距离&#xff0c;到包围它的元素的边框的距离&#xff1b; padding&#xff1a;内边距&#xff0c;用于控制内容与边框之间的距离&#xff1b; CSS padding&…

C 语言编程规范 -- 华为

1. 代码总体原则 1.1 清晰第一&#xff0c;清晰性是易于维护&#xff0c;易于重构的程序必须具备的特征 代码首先是给人读的&#xff0c;好的代码应当可以像文章一样发生朗诵出来&#xff0c;“程序必须为阅读它的人而编写&#xff0c;只是顺便用于机器执行” – Harold Abel…

视频集中存储/云存储平台EasyCVR国标GB28181协议接入的报文交互数据包分析

安防视频监控/视频集中存储/云存储/磁盘阵列EasyCVR平台可拓展性强、视频能力灵活、部署轻快&#xff0c;可支持的主流标准协议有国标GB28181、RTSP/Onvif、RTMP等&#xff0c;以及支持厂家私有协议与SDK接入&#xff0c;包括海康Ehome、海大宇等设备的SDK等。视频汇聚融合管理…

ChatGPT⼊门到精通(1):ChatGPT 是什么

⼀、直观感受 1、公司 OpenAI&#xff08;美国&#xff09; 2、官⽅⽹站 3、登录ChatGPT ![在这里插入图片描述](https://img-blog.csdnimg.cn/26901096553a4ba0a5c88c49b2601e6a.png 填⼊帐号、密码&#xff0c;点击登录。登录成功&#xff0c;如下 3、和ChatGPT对话 开始…

k8s之Pod及Probe 探针机制(健康检查机制)

文章目录 1、Pod1.1、定义1.2、Pod的形式1.3、Pod的使用1.4、 Pod生命周期1.5、初始化容器1.6、临时容器1.6.1、定义1.6.2、使用临时容器的步骤 1.7、静态Pod1.8、创建带标签的pod1.9、容器生命周期回调1.10、容器镜像使用秘钥从私有仓库下载1.11、多容器协同工作 2、Probe 探针…

GAN原理 代码解读

模型架构 代码 数据准备 import os import time import matplotlib.pyplot as plt import numpy as np import torchvision.transforms as transforms from torch.utils.data import DataLoader from torchvision import datasets import torch.nn as nn import torch# 创建文…

搭建 Qt6 开发环境

作者&#xff1a; 一去、二三里 个人微信号&#xff1a; iwaleon 微信公众号&#xff1a; 高效程序员 Qt 是一个跨平台的 C 应用程序开发框架&#xff0c;它提供了丰富的组件库和工具&#xff0c;使得开发人员可以在各种平台上轻松地开发 GUI 应用程序。 由于我们的教程 《细说…

CnetSDK .NET OCR SDK Crack

CnetSDK .NET OCR SDK Crack CnetSDK.NET OCR库SDK是一款高度准确的.NET OCR扫描仪软件&#xff0c;用于使用手写、文本和其他符号等图像进行字符识别。它是一款.NET OCR库软件&#xff0c;使用Tesseract OCR引擎技术&#xff0c;可将字符识别准确率提高99%。通过将此.NET OCR扫…

Rancher使用cert-manager安装报错解决

报错&#xff1a; rancher-rke-01:~/rke/rancher-helm/rancher # helm install rancher rancher-stable/rancher --namespace cattle-system --set hostnamewww.rancher.local Error: INSTALLATION FAILED: Internal error occurred: failed calling webhook "webhook…

sentinel的基本使用

在一些互联网项目中高并发的场景很多&#xff0c;瞬间流量很大&#xff0c;会导致我们服务不可用。 sentinel则可以保证我们服务的正常运行&#xff0c;提供限流、熔断、降级等方法来实现 一.限流&#xff1a; 1.导入坐标 <dependency><groupId>com.alibaba.c…

快速排序三种思路详解!

一、快速排序的介绍 快速排序是Hoare于1962年提出的一种二叉树结构的交换排序方法&#xff0c;其基本思想为&#xff1a;任取待排序元素序列中 的某元素作为基准值&#xff0c;按照该排序码将待排序集合分割成两子序列&#xff0c;左子序列中所有元素均小于基准值&#xff0c;…

激活函数总结(十九):激活函数补充(ISRU、ISRLU)

激活函数总结&#xff08;十九&#xff09;&#xff1a;激活函数补充 1 引言2 激活函数2.1 Inverse Square Root Unit &#xff08;ISRU&#xff09;激活函数2.2 Inverse Square Root Linear Unit (ISRLU)激活函数 3. 总结 1 引言 在前面的文章中已经介绍了介绍了一系列激活函…

python AI绘图教程

前提 1.安装python 2.安装git 步骤 下载stable-diffusion-webui项目&#xff08;链接&#xff1a;GitHub - AUTOMATIC1111/stable-diffusion-webui: Stable Diffusion web UI&#xff09; git clone https://github.com/AUTOMATIC1111/stable-diffusion-webui.git 安装st…

商城-学习整理-高级-消息队列(十七)

目录 一、RabbitMQ简介(消息中间件)1、RabbitMQ简介&#xff1a;2、核心概念1、Message2、Publisher3、Exchange4、Queue5、Binding6、Connection7、Channel8、Consumer9、Virtual Host10、Broker 二、一些概念1、异步处理2、应用解耦3、流量控制5、概述 三、Docker安装RabbitM…

【C++ 学习 ⑰】- 继承(下)

目录 一、派生类的默认成员函数 二、继承与友元 三、继承与静态成员 四、复杂的菱形继承及菱形虚拟继承 五、继承和组合 一、派生类的默认成员函数 派生类的构造函数必须调用基类的构造函数初始化基类的那一部分成员。如果基类没有默认构造函数&#xff0c;那么必须在派生…

报错sql_mode=only_full_group_by

首发博客地址 https://blog.zysicyj.top/ 报错内容 ### The error may exist in file[D:\code\cppCode20221025\leader-system\target\classes\mapper\system\TJsonDataMapper.xml] ### The error may involve defaultParameterMap ### The error occurred while…