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

这篇具有很好参考价值的文章主要介绍了【计算机网络】日志与守护进程。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

日志

一般使用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_t
   struct 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,//2
  BIND_ERR,//3
  LISTEN_ERR,//4
  SETSID_ERR,//5
  OPEN_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,即可退出守护进程文章来源地址https://www.toymoban.com/news/detail-665464.html

完整代码

err.hpp(错误信息枚举)

#pragma once 

enum
{
  USAGE_ERR=1,
  SOCKET_ERR,//2
  BIND_ERR,//3
  LISTEN_ERR,//4
  SETSID_ERR,//5
  OPEN_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);

  } 


到了这里,关于【计算机网络】日志与守护进程的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【计算机网络八股】计算机网络(一)

    计算机网络体系可以大致分为一下三种,OSI七层模型、TCP/IP四层模型和五层模型。 OSI七层模型:大而全,但是比较复杂、而且是先有了理论模型,没有实际应用。 TCP/IP四层模型:是由实际应用发展总结出来的,从实质上讲,TCP/IP只有最上面三层,最下面一层没有什么具体内

    2024年02月11日
    浏览(47)
  • 计算机网络----计算机网络的基础

    目录 一.计算机网络的相关概念 二.计算机网络的功能 三.计算机网络的发展 四.计算机网络的组成 五.计算机网络的分类 六.计算机的性能指标 1.速率 2.带宽 3.吞吐量 4.时延 5.时延带宽积 6.往返时延RTT 7.利用率 七.计算机的分层结构 八.ISO/OSI参考模型 九.OSI参考模型与TCP/IP参考模

    2024年03月17日
    浏览(47)
  • 计算机网络——计算机网络体系结构

    1.1 概念 一般认为,计算机网络是一个将分散的,具有独立功能的计算机系统,通过通信设备与线路连接起来,由功能完善的软件实现资源共享的信息传递的系统,简而言之,计算机网络就是一些 互联的,自治的计算机系统的集合 1.2 组成 (1)从组成部分:由 硬件,软件,

    2024年02月15日
    浏览(52)
  • 第一章 计算机网络概述【计算机网络】

    2023-3-26 17:07:26 以下内容源自《【计算机网络】》 仅供学习交流使用 计算机网络 计算机网络(第8版) 谢希仁 编著 1.2.1 网络的网络 计算机网络〈简称为网络)由若干结点(node) R和连接这些结点的链路(link)组成。 1.2.2互联网基础结构发展的三个阶段 请读者注意以下两个意思相

    2024年02月13日
    浏览(48)
  • 计算机网络-笔记-第一章-计算机网络概述

    一、第一章——计算机网络概述 二、第二章——物理层 三、第三章——数据链路层 四、第四章——网络层 五、第五章——运输层 六、第六章——应用层 目录 一、第一章——计算机网络概述 1、因特网概述 (1)网络、互联网、因特网 (2)因特网发展的三个阶段 (3)因特

    2024年02月11日
    浏览(51)
  • 【计算机网络原理】第一章:计算机网络概述

    1、计算机网络的诞生 从技术范畴来看,计算机网络是计算机技术与通信技术相互融合的产物。 2、计算机网络的定义 计算机网络是互连的、自治的计算机的集合 1)自治: 互连的计算机系统彼此独立,不存在主从或者控制与被控制的关系。 2)互连: 利用通信链路连接相互独立

    2024年04月08日
    浏览(51)
  • 【计算机网络】—— 详解码元,传输速率的计算|网络奇缘系列|计算机网络

    🌈个人主页:  Aileen_0v0 🔥系列专栏:  一见倾心,再见倾城  ---  计算机网络~ 💫个人格言: \\\"没有罗马,那就自己创造罗马~\\\" 目录 码元  速率和波特 思考1   思考2  思考3 带宽(Bandwidth)  📝总结 码元 是指用一个 固定时长的信号波形 _(数字脉冲),代表不同离散数值的基本波

    2024年02月04日
    浏览(61)
  • 【计算机网络】第一章、计算机网络体系结构

    1.计算机网络的组成 从不同的角度来看 内容 从组成上看 硬件、软件、协议 从工作方式上岸 边缘部分、核心部分 从功能上看 通信子网、资源子网 2.计算机网络的分类 角度 内容 分布范围 广域网、城域网、局域网、个域网 传输技术 广播式网络、点对点网络 拓扑结构 总线型

    2024年02月07日
    浏览(55)
  • 初识计算机网络 | 计算机网络的发展 | 协议初识

    1.计算机网络的发展 “矛盾是普遍存在的,矛盾是事物联系的实质内容和 事物发展的根本动力 !” 计算机在诞生之初,在军事上用来计算导弹的弹道轨迹!在发展的过程中( 商业的推动,国家政策推动 ),计算机的应用场景被发现!被应用于各种各样的场景当中! 现在,

    2024年01月25日
    浏览(50)
  • 【计算机网络】——前言计算机网络发展的历程概述

     ========================================================================= 主页点击直达: 个人主页 我的小仓库: 代码仓库 C语言偷着笑: C语言专栏 数据结构挨打小记: 初阶数据结构专栏 Linux被操作记: Linux专栏 LeetCode刷题掉发记: LeetCode刷题 算法: 算法专栏  C++头疼记: C++专栏 计算

    2024年02月08日
    浏览(55)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包