(C++通讯架构学习笔记):日志打印、优化main函数调用顺序

目录

基础设施之日志打印

设置时区

日志等级划分

捋顺main函数中代码执行顺序

基础设施之日志打印

  • 日志的重要性:供日后运行维护人员去查看、定位和解决问题
  • ngx_printf.cxx:放和打印格式相关的函数
  • ngx_log.cxx:放和日志相关的函数

【ngx_log_stderr()  】

  • ngx_log_stderr()  :三个特殊文件描述符,谈到了标准错误 STDERR_FILENO,代表屏幕
  • ngx_log_stderr():往屏幕上打印一条错误信息,功能类似于printf
    • printf("mystring=%s,myint=%d,%d","mytest",15,20)。
      • (1)根据可变的参数,组合出一个字符串:mystring=mytest,myint=15,20。
      • (2)往屏幕上显示出这个组合出来的字符串。
  • ngx_log_stderr():可以支持任意我们想支持的格式化字符 %d, %f,对于扩展原有功能非常有帮助。

【示例】

  • 函数调用关系
(i)void ngx_log_stderr(int err, const char *fmt, ...)
(i)    p = ngx_vslprintf(p,last,fmt,args); //实现了自我可定制的printf类似的功能
(i)        buf = ngx_sprintf_num(buf, last, ui64, zero, hex, width);
(i)    p = ngx_log_errno(p, last, err);
  • 在main函数中调用 

设置时区

  • 我们要设置成CST时区,以保证日期,时间显示的都正确
  • 我们常看到的时区,有如下几个:
    • PST【PST美国太平洋标准时间】 = GMT - 8;
    • GMT【格林尼治平均时间Greenwich Mean Time】等同于英国伦敦本地时间
    • UTC【通用协调时Universal Time Coordinated】 = GMT
    • CST【北京时间:北京时区是东八区,领先UTC八个小时】

【更改时区方式】

  • 输入命令:tzselect

  • 【注意】还需要拷贝生成的文件
cp /usr/share/zoneinfo/Asia/Shanghai  /etc/localtime

日志等级划分

  • 划分日志等级,一共分8级,分级的目的是方便管理,显示,过滤等等
  • 日志级别从高到低,数字最小的级别最高,数字最大的级别最低

【日志等级定义】

  • nginx.conf文件中进行配置

  • ngx_log.cxx
//描述:日志初始化,就是把日志文件打开 ,注意这里边涉及到释放的问题,如何解决?
void ngx_log_init()
{
    u_char *plogname = NULL;
    size_t nlen;

    //从配置文件中读取和日志相关的配置信息
    CConfig *p_config = CConfig::GetInstance();
    plogname = (u_char *)p_config->GetString("Log");
    if(plogname == NULL)
    {
        //没读到,就要给个缺省的路径文件名了
        plogname = (u_char *) NGX_ERROR_LOG_PATH; //"logs/error.log" ,logs目录需要提前建立出来
    }
    ngx_log.log_level = p_config->GetIntDefault("LogLevel",NGX_LOG_NOTICE);//缺省日志等级为6【注意】 ,如果读失败,就给缺省日志等级
    //nlen = strlen((const char *)plogname);

    //只写打开|追加到末尾|文件不存在则创建【这个需要跟第三参数指定文件访问权限】
    //mode = 0644:文件访问权限, 6: 110    , 4: 100:     【用户:读写, 用户所在组:读,其他:读】 
    ngx_log.fd = open((const char *)plogname,O_WRONLY|O_APPEND|O_CREAT,0644);  
    if (ngx_log.fd == -1)  //如果有错误,则直接定位到 标准错误上去 
    {
        ngx_log_stderr(errno,"[alert] could not open error log file: open() \"%s\" failed", plogname);
        ngx_log.fd = STDERR_FILENO; //直接定位到标准错误去了        
    } 
    return;
}
  • main函数中进行调用

  • nginx_global.h

  • nginx.cxx

捋顺main函数中代码执行顺序

//本文件用的函数声明
static void freeresource();

//和设置标题有关的全局量
char **g_os_argv;            //原始命令行参数数组,在main中会被赋值
char *gp_envmem = NULL;      //指向自己分配的env环境变量的内存,在ngx_init_setproctitle()函数中会被分配内存
int  g_environlen = 0;       //环境变量所占内存大小

//和进程本身有关的全局量
pid_t ngx_pid;               //当前进程的pid

int main(int argc, char *const *argv)
{   
    int exitcode = 0;           //退出代码,先给0表示正常退出

    //(1)无伤大雅也不需要释放的放最上边    
    ngx_pid = getpid();         //取得进程pid
    g_os_argv = (char **) argv; //保存参数指针    

    //(2)初始化失败,就要直接退出的
    //配置文件必须最先要,后边初始化啥的都用,所以先把配置读出来,供后续使用 
    CConfig *p_config = CConfig::GetInstance(); //单例类
    if(p_config->Load("nginx.conf") == false) //把配置文件内容载入到内存        
    {        
        ngx_log_stderr(0,"配置文件[%s]载入失败,退出!","nginx.conf");
        //exit(1);终止进程,在main中出现和return效果一样 ,exit(0)表示程序正常, exit(1)/exit(-1)表示程序异常退出,exit(2)表示表示系统找不到指定的文件
        exitcode = 2; //标记找不到文件
        goto lblexit;
    }
    
    //(3)一些初始化函数,准备放这里
    ngx_log_init();             //日志初始化(创建/打开日志文件)


    //(4)一些不好归类的其他类别的代码,准备放这里
    ngx_init_setproctitle();    //把环境变量搬家

    
    //--------------------------------------------------------------    
    for(;;)
    //for(int i = 0; i < 10;++i)
    {
        sleep(1); //休息1秒        
        printf("休息1秒\n");        

    }
      
    //--------------------------------------
lblexit:
    //(5)该释放的资源要释放掉
    freeresource();  //一系列的main返回前的释放动作函数
    printf("程序退出,再见!\n");
    return exitcode;
}

//专门在程序执行末尾释放资源的函数【一系列的main返回前的释放动作函数】
void freeresource()
{
    //(1)对于因为设置可执行程序标题导致的环境变量分配的内存,我们应该释放
    if(gp_envmem)
    {
        delete []gp_envmem;
        gp_envmem = NULL;
    }

    //(2)关闭日志文件
    if(ngx_log.fd != STDERR_FILENO && ngx_log.fd != -1)  
    {        
        close(ngx_log.fd); //不用判断结果了
        ngx_log.fd = -1; //标记下,防止被再次close吧        
    }
}

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
这是一门linux下c++通讯架构实战课程,针对c/c++语言已经掌握的很熟并希望进一步深造以将来用c++在linux下从事网络通讯领域/网络服务器的开发和架构工作。这门课程学习难度颇高但也有着极其优渥的薪水(最少30K月薪,最高可达60-80K月薪),这门课程,会先从nginx源码的分析和讲解开始,逐步开始书写属于自己的高性能服务器框架代码,完善个人代码库,这些,将会是您日后能取得高薪的重要筹码。本课程原计划带着大家逐行写代码,但因为代码实在过于复杂和精细,带着写代码可能会造成每节课至少要4~5小时的超长时间,所以老师会在课前先写好代码,主要的时间花费在逐行讲解这些代码上,这一点望同学们周知。如果你觉得非要老师领着写代码才行的话,老师会觉得你当前可能学习本门课程会比较吃力,请不要购买本课程,以免听不懂课程并给老师差评,差评也会非常影响老师课程的销售并造成其他同学的误解。 这门课程要求您具备下面的技能:(1)对c/c++语言掌握的非常熟练,语言本身已经不是继续学习的障碍,并不要求您一定熟悉网络或者linux;(2)对网络通讯架构领域有兴趣、勇于挑战这个高难度的开发领域并期望用大量的付出换取高薪;在这门课程中,实现了一个完整的项目,其中包括通讯框架和业务逻辑框架,浓缩总结起来包括如下几点:(1)项目本身是一个极完整的多线程高并发的服务器程序;(2)按照包头包体格式正确的接收客户端发送过来的数据包, 完美解决收包时的数据粘包问题;(3)根据收到的包的不同来执行不同的业务处理逻辑;(4)把业务处理产生的结果数据包正确返回给客户端;本项目用到的主要开发技术和特色包括:(1)epoll高并发通讯技术,用到的触发模式是epoll中的水平触发模式【LT】;(2)自己写了一套线程池来处理业务逻辑,调用适当的业务逻辑处理函数处理业务并返回给客户端处理结果;(3)线程之间的同步技术包括互斥量,信号量等等;(4)连接池中连接的延迟回收技术,这是整个项目中的精华技术,极大程度上消除诸多导致服务器程序工作不稳定的因素;(5)专门处理数据发送的一整套数据发送逻辑以及对应的发送线程;(6)其他次要技术,包括信号、日志打印、fork()子进程、守护进程等等;
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值