glog 快速使用教程

场景

1.大部分程序由函数组成, 每个函数执行一段设计好的逻辑, 但是大部分的时候有可能出现意料之外的值, 这时候就很想知道这种意料以外的值是如何产生的, 这就需要一个函数调用和参数跟踪, 日志正好发挥作用.

2.调试多线程程序时某些函数执行顺序可能出现期望之外的顺序,或者需要知道出问题时线程执行的顺序,而Debug程序并不能同时Debug多个线程的函数调用, 而且会影响线程的竞争导致bug无法重现,这时只能用日志.

3.在客户使用软件时, 经常需要分析用户的使用习惯和行为, 从而根据用户行为习惯来重点推荐产品, 这时候就需要历史日志.

说明

1.glog只支持utf8编码的日志文件.

2.glog的严重程度级别只支持4种, INFO, WARNING, ERROR, and FATAL. 也就是LOG(XXX); DLOG用来记录DEBUG模式的日志, 在Release模式下会未定义,不用担心性能问题(#ifndef NDEBUG). 默认情况下会把 ERROR 或 FATAL 级别输出到stderr.

3.条件或偶发性日志, 可允许指定变量满足特定值才输出日志. 比如a>10,每请求10次日志才输出一次,限制输出的前几次. LOG_IF,LOG_EVERY_N,LOG_IF_EVERY_N,LOG_FIRST_N. 如果只需要在Debug模式下输出. DLOG,DLOG_IF,DLOG_EVERY_N.

4.检查宏用来检查期待的条件, 如果不满足条件,就会中断程序.CHECK,CHECK_NE,CHECK_EQ,CHECK_EQ,CHECK_NOTNULL,CHECK_STREQ,CHECK_STRNE,CHECK_STRCASEEQ,CHECK_STRCASENE 等等. 如果不想让程序中断的话, 这些宏很少用.

5.设置标记, 可以通过命令行设置参数, 比如 ./your_application –logtostderr=1, 但是要使用 Google gflags library. 如果没有使用gflags库, 可以使用环境变量设置. GLOG_logtostderr=1, FLAGS_v=1. 或者在代码里设置 FLAGS_*.

6.Raw Logging, 使用头文件

例子

  1. 注意, 这个例子只是产生输出, 如果想把日志输出到文件, 那么把 FLAGS_logtostderr=false.
// test-log.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"

// 使用Windows自带的SEVERITIES定义
#define GLOG_NO_ABBREVIATED_SEVERITIES
#include <Windows.h>

#include <glog/logging.h>
#include <glog/raw_logging.h>

static char* ConvertUnicodeToUtf8(const wchar_t* unicode)
{
    if(unicode == NULL)
    {
        return strdup("\0");
    }

    int len;
    len = WideCharToMultiByte(CP_UTF8, 0,unicode, -1, NULL, 0, NULL, NULL);
    char *szUtf8 = (char*)malloc(len + 1);
    memset(szUtf8, 0, len + 1);
    WideCharToMultiByte(CP_UTF8, 0,unicode, -1, szUtf8, len, NULL,NULL);
    return szUtf8;
}

void TestLogSeverityLevel()
{
    LOG(INFO) << "TestLogSeverityLevel the INFO text";
    DLOG(INFO) << "TestLogSeverityLevel the DEBUG text";
    LOG(ERROR) << "TestLogSeverityLevel the ERROR text1";
    LOG(WARNING) << "TestLogSeverityLevelthe warning text";
    LOG(ERROR) << "TestLogSeverityLevel the ERROR text2";
    // DLOG(FATAL) << "DEBUG FATAL"; // 会产生一个abort中断之前先输出到日志
}

void TestSettingFlags()
{
    // 日志文件格式
    ///tmp/<program name>.<hostname>.<user name>.log.<severity level>.<date>.<time>.<pid>
    // test-log.exe.APPLE-PC.apple.log.INFO.20161230-155337.2472

    // 注意,文件路径编码必须是GB2312,不然不识别以下中文路径,如果文件不是本地编码(ANSI),那么需要转换为ANSI编码.
    // 设置日志目录,必须是ANSI(中文系统即GB2312)编码,不支持UNICODE
    // 该目录必须存在,不然生成不了日志文件.
    FLAGS_logtostderr = true; // 不输出到文件,输出到stderr.
    FLAGS_log_dir = "E:\\新建文件夹\\";
    FLAGS_max_log_size = 20; // 设置最大日志文件大小.
}

void TestConditionalAndOccasionalLogging()
{
    for(int i = 0; i< 100;++i)
    {
        LOG_IF(INFO,i%10 == 0) << "LOG_IF i is " << i; // 每10次输出一次.
        LOG_EVERY_N(INFO,10) << " LOG_EVERY_N i is " << i <<  " Got the " << google::COUNTER; // 也是每间隔10次输出
        LOG_FIRST_N(INFO, 20) << "LOG_FIRST_N i is " << i << " Got the " << google::COUNTER; // 输出前20个
    }
}

void TestCHECKMacros()
{
    std::string first = "abcd";
    std::string second = "AbCd";
    CHECK_EQ(first,second); //产生一个中断之前先输出到日志
}

int _tmain(int argc, _TCHAR* argv[])
{
    // Initialize Google's logging library.
    // glog保存的文件时UTF-8格式编码,所以需要转码需要记录的字符串为UTF-8,转换为utf8.

    char* argv0 = ConvertUnicodeToUtf8(argv[0]);
    google::InitGoogleLogging(argv0);

    TestSettingFlags();
    TestLogSeverityLevel();
    TestConditionalAndOccasionalLogging();
    TestCHECKMacros();

    // 日志内容
    //Log file created at: 2016/12/30 15:53:37
    //Running on machine: APPLE-PC
    //Log line format: [IWEF]mmdd hh:mm:ss.uuuuuu threadid file:line] msg
    //I1230 15:53:37.337400  5456 test-log.cpp:35] the same file

    //google::FlushLogFiles(google::GLOG_ERROR);
    google::ShutdownGoogleLogging();
    return 0;
}

输出

I1231 20:33:15.303700  5852 test-log.cpp:31] TestLogSeverityLevel the INFO text
I1231 20:33:15.303700  5852 test-log.cpp:32] TestLogSeverityLevel the DEBUG text

E1231 20:33:15.303700  5852 test-log.cpp:33] TestLogSeverityLevel the ERROR text
1
W1231 20:33:15.303700  5852 test-log.cpp:34] TestLogSeverityLevelthe warning tex
t
E1231 20:33:15.303700  5852 test-log.cpp:35] TestLogSeverityLevel the ERROR text
2
I1231 20:33:15.303700  5852 test-log.cpp:57] LOG_IF i is 0
I1231 20:33:15.303700  5852 test-log.cpp:58]  LOG_EVERY_N i is 0 Got the 1
I1231 20:33:15.303700  5852 test-log.cpp:59] LOG_FIRST_N i is 0 Got the 1
I1231 20:33:15.303700  5852 test-log.cpp:59] LOG_FIRST_N i is 1 Got the 2
I1231 20:33:15.303700  5852 test-log.cpp:59] LOG_FIRST_N i is 2 Got the 3
I1231 20:33:15.303700  5852 test-log.cpp:59] LOG_FIRST_N i is 3 Got the 4
I1231 20:33:15.303700  5852 test-log.cpp:59] LOG_FIRST_N i is 4 Got the 5
I1231 20:33:15.303700  5852 test-log.cpp:59] LOG_FIRST_N i is 5 Got the 6
I1231 20:33:15.303700  5852 test-log.cpp:59] LOG_FIRST_N i is 6 Got the 7
I1231 20:33:15.303700  5852 test-log.cpp:59] LOG_FIRST_N i is 7 Got the 8
I1231 20:33:15.303700  5852 test-log.cpp:59] LOG_FIRST_N i is 8 Got the 9
I1231 20:33:15.303700  5852 test-log.cpp:59] LOG_FIRST_N i is 9 Got the 10
I1231 20:33:15.303700  5852 test-log.cpp:57] LOG_IF i is 10
I1231 20:33:15.303700  5852 test-log.cpp:58]  LOG_EVERY_N i is 10 Got the 11
I1231 20:33:15.303700  5852 test-log.cpp:59] LOG_FIRST_N i is 10 Got the 11
I1231 20:33:15.319300  5852 test-log.cpp:59] LOG_FIRST_N i is 11 Got the 12
I1231 20:33:15.319300  5852 test-log.cpp:59] LOG_FIRST_N i is 12 Got the 13
I1231 20:33:15.319300  5852 test-log.cpp:59] LOG_FIRST_N i is 13 Got the 14
I1231 20:33:15.319300  5852 test-log.cpp:59] LOG_FIRST_N i is 14 Got the 15
I1231 20:33:15.319300  5852 test-log.cpp:59] LOG_FIRST_N i is 15 Got the 16
I1231 20:33:15.319300  5852 test-log.cpp:59] LOG_FIRST_N i is 16 Got the 17
I1231 20:33:15.319300  5852 test-log.cpp:59] LOG_FIRST_N i is 17 Got the 18
I1231 20:33:15.319300  5852 test-log.cpp:59] LOG_FIRST_N i is 18 Got the 19
I1231 20:33:15.319300  5852 test-log.cpp:59] LOG_FIRST_N i is 19 Got the 20
I1231 20:33:15.319300  5852 test-log.cpp:57] LOG_IF i is 20
I1231 20:33:15.319300  5852 test-log.cpp:58]  LOG_EVERY_N i is 20 Got the 21
I1231 20:33:15.319300  5852 test-log.cpp:57] LOG_IF i is 30
I1231 20:33:15.319300  5852 test-log.cpp:58]  LOG_EVERY_N i is 30 Got the 31
I1231 20:33:15.319300  5852 test-log.cpp:57] LOG_IF i is 40
I1231 20:33:15.319300  5852 test-log.cpp:58]  LOG_EVERY_N i is 40 Got the 41
I1231 20:33:15.319300  5852 test-log.cpp:57] LOG_IF i is 50
I1231 20:33:15.319300  5852 test-log.cpp:58]  LOG_EVERY_N i is 50 Got the 51
I1231 20:33:15.319300  5852 test-log.cpp:57] LOG_IF i is 60
I1231 20:33:15.319300  5852 test-log.cpp:58]  LOG_EVERY_N i is 60 Got the 61
I1231 20:33:15.319300  5852 test-log.cpp:57] LOG_IF i is 70
I1231 20:33:15.319300  5852 test-log.cpp:58]  LOG_EVERY_N i is 70 Got the 71
I1231 20:33:15.319300  5852 test-log.cpp:57] LOG_IF i is 80
I1231 20:33:15.319300  5852 test-log.cpp:58]  LOG_EVERY_N i is 80 Got the 81
I1231 20:33:15.319300  5852 test-log.cpp:57] LOG_IF i is 90
I1231 20:33:15.319300  5852 test-log.cpp:58]  LOG_EVERY_N i is 90 Got the 91
F1231 20:33:15.319300  5852 test-log.cpp:67] Check failed: first == second (abcd
 vs. AbCd)
*** Check failure stack trace: ***

参考

1.源码的doc/glog.html

[glog]_[C/C++]_[使用glog来记录日志]的更多相关文章

  1. Spring_MVC_教程_快速入门_深入分析

    Spring MVC 教程,快速入门,深入分析 博客分类: SPRING Spring MVC 教程快速入门  资源下载: Spring_MVC_教程_快速入门_深入分析V1.1.pdf Spring ...

  2. 基于samba实现win7与linux之间共享文件_阳仔_新浪博客

    基于samba实现win7与linux之间共享文件_阳仔_新浪博客 然后启动samba执行如下指令: /dev/init.d/smb start 至此完成全部配置.

  3. 浅谈 Underscore.js 中 _.throttle 和 _.debounce 的差异

    Underscore.js是一个很精干的库,压缩后只有5.2KB.它提供了几十种函数式编程的方法,弥补了标准库的不足,大大方便了JavaScript的编程. 本文仅探讨Underscore.js的两个 ...

  4. CLOSE-UP FORMALWEAR_意大利进口_2015秋冬_男装发布会_西装图片系列_男装西装设计资料_WeArTrends时尚资讯网_国内最专业的服装设计资讯网站

    CLOSE-UP FORMALWEAR_意大利进口_2015秋冬_男装发布会_西装图片系列_男装西装设计资料_WeArTrends时尚资讯网_国内最专业的服装设计资讯网站 CLOSE-UP FORMA ...

  5. 聚焦设计交易与商业落地 DANG·DHUB设计师平台上线【图】_品牌资讯_服饰_太平洋时尚网

    聚焦设计交易与商业落地 DANG·DHUB设计师平台上线[图]_品牌资讯_服饰_太平洋时尚网 聚焦设计交易与商业落地 DANG·DHUB设计师平台上线

  6. 联系我们_你我想法_【有男度】UNANDU 100%进口 全球设计师品牌精汇 男装_男装搭配_时尚男装_品牌男装_男装搭配技巧_男装网站

    联系我们_你我想法_[有男度]UNANDU 100%进口 全球设计师品牌精汇 男装_男装搭配_时尚男装_品牌男装_男装搭配技巧_男装网站 联系我们 2012-02-17   国内北京公司总部  邮编 ...

  7. Payssion,海外本地支付_海外本地收款_小语种本地支付_外贸收款_外贸网店收款_欧洲本地支付_俄罗斯本地支付_巴西支付_跨境支付_PAYSSION,让跨境支付更轻松!

    Payssion,海外本地支付_海外本地收款_小语种本地支付_外贸收款_外贸网店收款_欧洲本地支付_俄罗斯本地支付_巴西支付_跨境支付_PAYSSION,让跨境支付更轻松!       首页 / 关于 ...

  8. 北京西服定做_衬衫定制_关于我们_Dimoon TLR.

    北京西服定做_衬衫定制_关于我们_Dimoon TLR.

  9. MUD江湖_MUD文字游戏_MUD五指_武林群侠_北侠_夺宝江湖_书剑_文字江湖游戏_MUD游戏下载

    MUD江湖_MUD文字游戏_MUD五指_武林群侠_北侠_夺宝江湖_书剑_文字江湖游戏_MUD游戏下载  武侠类手机文字游戏,经典再现高度自由玩法宠物 自制装备 师徒自立门派 自造武功欢迎来玩 Q群 1 ...

  10. 轻奢请向历史SAY NO_重青网_重庆青年报_重庆青年报电子版_重庆青年报网站_重庆青年报官方网站

    轻奢请向历史SAY NO_重青网_重庆青年报_重庆青年报电子版_重庆青年报网站_重庆青年报官方网站 轻奢请向历史SAY NO 经济学家George Taylor在他著名的"裙摆指数" ...

随机推荐

  1. Oracle EBS 跳跳转标准销售订单程序转标准销售订单程序

    -- 打开PO PROCEDURE Btn_Open_Po IS BEGIN Fnd_Function.Execute(Function_Name => 'PO_POXPOEPO', Open_ ...

  2. 【gp数据库】OLTP和OLAP区别详解

    原来一直使用Oracle,新公司使用greenplum后发现系统的并发性差很多,后来才了解因为Oracle属于OLTP类型,而gp数据库属于OLAP类型的.具体了解如下: 数据库系统一般分为两种类型, ...

  3. NodeJS做中转服务器,转发接口

    搬家后的博客地址:http://www.cnblogs.com/shihaibin821/p/7683752.html

  4. Nginx 泛解析配置请求映射到多端口实现二级域名访问

    由于想实现一个域名放置多个应用运行的目的,而不想通过域名后加端口号方式处理,这种方式处理记起来太麻烦,偷懒党简直不能忍,故而考虑了使用二级域名来处理多个应用同时运行.Google了一番资料并进行了尝试 ...

  5. November 25th 2016 Week 48th Friday

    People will fall for its appearance while driving passionately. 观者倾心,驭者动魄. This is an advertisement ...

  6. MFC调试时可以,使用生产的exe时,显示未响应解决方案

    首先,使用release模式,生成解决方案和exe. 对生成的exe,选择属性,兼容性,win7,应用,即可解决. 此方法对debug无用

  7. 鼠标有但是U盘读取不出来怎么办

    我今天就遇到了这个问题,搞了半天最后下了一个驱动人生,查看里面的回答才解决 就是把里面通用串行总控制器设置为隐藏文件可读之后选择把灰色的都删除就好了.具体可以在驱动人生里搜U盘不识别,之后就4,5步即 ...

  8. 026.2 网络编程 UDP聊天

    实现,通过socket对象 ##############################################################需求建立UDP发送端:###思路:1.建立可以实 ...

  9. mysql 索引分类以及用途分析

    MySQL索引分为普通索引.唯一性索引.全文索引.单列索引.多列索引等等.这里将为大家介绍着几种索引各自的用途. 一. MySQL: 索引以B树格式保存 Memory存储引擎可以选择Hash或BTre ...

  10. Css中路径data用法

    Data URI scheme是在RFC2397中定义的,目的是将一些小的数据,直接嵌入到网页中,从而不用再从外部文件载入. data:,文本数据 data:text/plain,文本数据 data: ...