项目中需要打log,当初看到glog,觉得google出品应该差不了,而且简单易用,库不是很大,就选择他了。

但是在使用中还真的发现一些不顺手和库设计上的问题,反正和我的使用习惯有点不一样。

  1. 设置log参数时有的用google::xx以函数的形式设置,有的以FLAG_xx的形式设置,而且有一些设置项两种方式都可以,但是源码中又走的不是相互封装的关系,看着有点乱,没理解glog为什么要这么设计,为什么不统一用一种方式。
  2. 在使用时我一直以为没有调用google::InitGoogleLogging是不会打印任何log的,直到有用户跟我反映说,他的程序里控制台一直会输出log信息,我明明把输出到控制台的选项都关闭了啊
    FLAGS_logtostderr = 0;	//是否打印到控制台
    FLAGS_alsologtostderr = 0; //打印到日志同时是否打印到控制台

      (这里也要吐槽下,不明白为什么要有FLAGS_alsologtostderr,输出到控制台和输出到日志文件分别有一个变量控制不行么,两个变量控制同一个状态真的好么)。后来看了glog源码才发现google::InitGoogleLogging根本不控制什么东西啊,

    void InitGoogleLogging(const char* argv0) {
    glog_internal_namespace_::InitGoogleLoggingUtilities(argv0);
    } void InitGoogleLoggingUtilities(const char* argv0) {
    CHECK(!IsGoogleLoggingInitialized())
    << "You called InitGoogleLogging() twice!";
    const char* slash = strrchr(argv0, '/');
    #ifdef OS_WINDOWS
    if (!slash) slash = strrchr(argv0, '\\');
    #endif
    g_program_invocation_short_name = slash ? slash + 1 : argv0;
    g_main_thread_id = pthread_self(); #ifdef HAVE_STACKTRACE
    InstallFailureFunction(&DumpStackTraceAndExit);
    #endif
    }

      查了一下g_program_invocation_short_name,g_main_thread_id这两个变量只是获取一下信息为了防止重复初始化和打印时输出线程ID,根本不控制是否可以输出。不信再看下面

    因为glog都是通过LOG(XXX)来打印log,所以找到源码

    #define LOG(severity) COMPACT_GOOGLE_LOG_ ## severity.stream()
    ----------------------------------------------------
    #if GOOGLE_STRIP_LOG == 0
    #define COMPACT_GOOGLE_LOG_INFO google::LogMessage( \
    __FILE__, __LINE__)
    #define LOG_TO_STRING_INFO(message) google::LogMessage( \
    __FILE__, __LINE__, google::GLOG_INFO, message)
    #else
    #define COMPACT_GOOGLE_LOG_INFO google::NullStream()
    #define LOG_TO_STRING_INFO(message) google::NullStream()
    #endif
    ----------------------------------------------------
    LogMessage::LogMessage(const char* file, int line, LogSeverity severity,
    int ctr, void (LogMessage::*send_method)())
    : allocated_(NULL) {
    Init(file, line, severity, send_method);
    data_->stream_.set_ctr(ctr);
    } ----------------------------------------------------
    void LogMessage::Init(const char* file,
    int line,
    LogSeverity severity,
    void (LogMessage::*send_method)()) {
    allocated_ = NULL;
    if (severity != GLOG_FATAL || !exit_on_dfatal) {
    allocated_ = new LogMessageData();
    data_ = allocated_;
    data_->first_fatal_ = false;
    } else {
    MutexLock l(&fatal_msg_lock);
    if (fatal_msg_exclusive) {
    fatal_msg_exclusive = false;
    data_ = &fatal_msg_data_exclusive;
    data_->first_fatal_ = true;
    } else {
    data_ = &fatal_msg_data_shared;
    data_->first_fatal_ = false;
    }
    } stream().fill('0');
    data_->preserved_errno_ = errno;
    data_->severity_ = severity;
    data_->line_ = 0;
    data_->send_method_ = send_method;
    data_->sink_ = NULL;
    data_->outvec_ = NULL;
    WallTime now = WallTime_Now();
    data_->timestamp_ = static_cast<time_t>(now);
    localtime_r(&data_->timestamp_, &data_->tm_time_);
    int usecs = static_cast<int>((now - data_->timestamp_) * 1000000);
    RawLog__SetLastTime(data_->tm_time_, usecs); data_->num_chars_to_log_ = 0;
    data_->num_chars_to_syslog_ = 0;
    data_->basename_ = const_basename("");
    data_->fullname_ = file;
    data_->has_been_flushed_ = false; // If specified, prepend a prefix to each line. For example:
    // I1018 160715 f5d4fbb0 logging.cc:1153]
    // (log level, GMT month, date, time, thread_id, file basename, line)
    // We exclude the thread_id for the default thread.
    if (FLAGS_log_prefix && (line != kNoLogPrefix)) {
    stream() << LogSeverityNames[severity][0]
    << setw(2) << 1+data_->tm_time_.tm_mon
    << setw(2) << data_->tm_time_.tm_mday
    << ' '
    << setw(2) << data_->tm_time_.tm_hour << ':'
    << setw(2) << data_->tm_time_.tm_min << ':'
    << setw(2) << data_->tm_time_.tm_sec << "."
    << setw(6) << usecs
    << ' '
    << setfill(' ') << setw(5)
    << static_cast<unsigned int>(GetTID()) << setfill('0')
    << ' '
    << data_->basename_ << ':' << data_->line_ << "] ";
    }
    data_->num_prefix_chars_ = data_->stream_.pcount(); if (!FLAGS_log_backtrace_at.empty()) {
    char fileline[128];
    snprintf(fileline, sizeof(fileline), "%s:%d", data_->basename_, line);
    #ifdef HAVE_STACKTRACE
    if (!strcmp(FLAGS_log_backtrace_at.c_str(), fileline)) {
    string stacktrace;
    DumpStackTraceToString(&stacktrace);
    stream() << " (stacktrace:\n" << stacktrace << ") ";
    }
    #endif
    }
    }

      看到没,其实都在LogMessage::Init中,执行了这么多代码你有看到g_program_invocation_short_name,g_main_thread_id控制是否输出么,也就是说就算你没调用google::InitGoogleLogging,你执行LOG(XXX)<<XX;也会执行这么多代码,在一个要求执行效率比较高的代码里,这会浪费很多cpu时间啊。

      到这我也明白了客户那边为什么会输出控制台log,因为我以为没调用google::InitGoogleLogging就不会执行任何glog函数,用户使用我的接口时没有调用google::InitGoogleLogging,但是接口里面直行到LOG(XXX)时还是会输出到控制台信息。

    但是还有一个问题,那就是为什么只输出到控制台,没有输出到log文件,那么接着来看

随便找一个函数看看
google::SetLogFilenameExtension(Extra_Name);
-------------------------------------------------------
void SetLogFilenameExtension(const char* ext) {
LogDestination::SetLogFilenameExtension(ext);
} 可以知道google::xxx都是通过LogDestination类来设置的,但是LogDestination类在哪里初始化呢,找找new LogDestination吧
------------------------------------------------------
inline LogDestination* LogDestination::log_destination(LogSeverity severity) {
assert(severity >= && severity < NUM_SEVERITIES);
if (!log_destinations_[severity]) {
log_destinations_[severity] = new LogDestination(severity, NULL);
}
return log_destinations_[severity];
}
可以看到,它的逻辑是使用severity等级来定义对象,每个等级对应一个LogDestination对象,有一个LogDestination的数组。其他函数也都一样,每次都是从这个数组中取出对象来设置。
-------------------------------------------------------

因为我把log的初始化封装到一个函数里,用户没有初始化,所以没有调用任何google::xx,或者FLAGS_XX代码,所以根据glog库的默认值,是不会输出到glog的,但是问题还是,我的接口中cpu还是会执行LOG(XXX),所以其实还是执行了很多glog库里的代码。

所以如果真的想不输出log,不能通过glog函数来控制,需要自己再封装一下,在自己的代码中控制,这个glog实在太差劲了,这太坑。

这里也证明了,有人问FLAGS_XX,google::xx,应该在google::InitGoogleLogging之前还是之后执行,回答是没关系,他俩根本不存在先后顺序。

3.    用户还提出一个需求,就是不希望log中显示文件名和行号,我一开始以为这太容易了,肯定会有个设置项设置的,后来发现我想的太美了,哪有这个设置。还是老老实实看源码吧

LogMessage::LogMessage(const char* file, int line, LogSeverity severity,
int ctr, void (LogMessage::*send_method)())
: allocated_(NULL) {
Init(file, line, severity, send_method);
data_->stream_.set_ctr(ctr);
}

前面我们知道了,输出log其实归根结底都是调用LogMessage,而且这两个参数const char* file, int line在从对外的接口函数里一路传下来,想不打印文件名和行号,只能自己改源码再重新编译库dll了,因为LogMessage是必经之路,所以我就是在LogMessage改动,把file,line给个固定值,算是解决了问题。

可是如果每次有点问题就需要改源码重新编译库文件,那也是够够的了,改来改去的,万一将来有一天需要升级最新的glog版本,难道我要再拿最新的源码把这些改动再改一边么,想想就够了。

所以啊,这第三方库的选择就跟去饭店吃饭一样,冷清的饭店肯定是有问题的,要么菜不好吃,要么服务不行,要么有什么其他问题,还是随大流比较好,小众的东西还是谨慎尝试,需要不拍踩坑才行。

关于glog使用中遇到的问题的更多相关文章

  1. ORA-02020 : 过多的数据库链接在使用中-Windows环境解决步骤

    一.现象 编译存储过程时报ORA-02020错误. 错误详细信息:ORA-04052在查找远程对象 xx@yy时出错 ORA-00604 : 递归 SQL 级别 1 出现错误 ORA-02020 : ...

  2. Subsonic使用中

    使用中,遇到各种奇葩问题,依依汇总. 1.引用了Subsonic层后,一运行就开始报错,提示未能找到文件!!    //引用后,目标框架可能会被改变,subsonic的默认框架是2.0,请检查框架是否 ...

  3. mysqldump 备份命令使用中的一些经验总结

    mysqldump的一个小坑(自测) 正文:经常使用接触mysql复制功能的朋友应该对mysqldump命令不陌生吧,鄙人最近也在研究学习这一块的内容,经过几天的测试,发现mysqldump使用中容易 ...

  4. <总结>delphi WebBrowser控件的使用中出现的bug

    Delphi WebBrowser控件的使用中出现的bug:  1.WebBrowser.Visible=false:Visible属性不能使WebBrowser控件不可见,暂时用 WebBrowse ...

  5. SSH框架使用中存在的诡异异常

    背景 相信大多数人目前都在使用Spring + Struts2/SpringMVC + Hibernate来构建项目的整体架构,但是在使用中经藏会遇到一些诡异的问题,不知道如果解决,今天我遇到了一个非 ...

  6. VM出现该虚拟机正在使用中的提示,让获取所有权限解决办法

    今天打开虚拟机正要学习,结果说是虚拟机似乎正在使用中,让我重新获取权限.解决办法:打开提示的配置文件的位置,将一个以.lck结尾的文件夹删除或者保存为另外的文件名称,再打开虚拟机就OK了.

  7. cordova + ionic 使用中碰到的一些问题

    cordova + ionic 使用中碰到的一些问题     No Content-Security-Policy meta tag found. Please add one when using ...

  8. VMware“该虚拟机似乎正在使用中”问题

    在用VMware虚拟机的时候,有时会发现打开虚拟机时提示“该虚拟机似乎正在使用中.如果该虚拟机未在使用,请按“获取所有权(T)”按钮获取它的所有权.否则,请按“取消(C)”按钮以防损坏.配置文件: D ...

  9. 项目使用中Linq使用总结

    项目使用中Linq使用总结 本文旨在和网友分享Linq在项目中的实践,曾经我参与过的项目都能看见Linq的影子.(LinqTosql.LinqToString.LinqToXML.LinqToEnti ...

随机推荐

  1. 提取全局应用程序集中的dll

    如何提取全局应用程序集中的dll GAC路径:C:\WINDOWS\assembly 在资源管理器中打开这个路径看到的东西不能复制,右键中也只有 “卸载” 和 “属性” 两个菜单. 在命令提示符下切换 ...

  2. ssh转发流量的四种姿势

    在很多时候拿到了内网的一台主机,我们需要用它做跳板来对内网进一步扩大战果. 也许方法很多,meterpreter,nc等等.但是最方便也最有可能穿透防火墙的方法,就是用ssh. 分为四种类型: 本地转 ...

  3. canvas画圆又毛边

    canvas使用arc()画园有毛边,如图:,只需给其添加width,height即可,直接上代码 <!DOCTYPE html> <html lang="en" ...

  4. pymysql操作数据库、索引、慢日志管理

    目录 pymysql操作数据库 简单操作 sql的注入问题 sql注入问题解决办法 sql注入问题模板总结 利用pymysql操作数据库 (增删改),conn.commit() 索引 1.为何要有索引 ...

  5. Python 练习实例3

    Python 练习实例3 题目:一个整数,它加上100后是一个完https://www.xuanhe.net/全平方数,再加上168又是一个完全平方数,请问该数是多少? 程序分析: 假设该数为 x. ...

  6. PHP大文件上传断点续传解决方案

    1.使用PHP的创始人 Rasmus Lerdorf 写的APC扩展模块来实现(http://pecl.php.net/package/apc) APC实现方法: 安装APC,参照官方文档安装,可以使 ...

  7. Visual Studio 2008:路径设置

    造冰箱的大熊猫,本文适用于Visual Studio 2008中文版@cnblogs 2018/11/30 1.头文件路径设置 如果头文件所在路径未在环境变量中定义,编译时会出现C1083错误,提示无 ...

  8. UVa 1595 Symmetry (set && math)

    题意:给出n个在直角坐标系上的点,问你能不能找出一条竖轴(即垂直于x的轴)使得所有的点根据这条轴对称,能则输出YES,否则输出NO 分析:首先需要找到对称轴的值,将所有n个点的x轴的值加起来然后去除以 ...

  9. 分布式-信息方式-ActiveMQ的Destination高级特性1

    ActiveMQ的Destination高级特性 Destination高级特性----->Composite Destinations 组合队列Composite Destinations : ...

  10. Spring boot之Hello World访问404

    (1)404 -- 确定地址是否输入正确,,此路径非彼路径 (2)404 -- 是否用对注解,此注解非彼注解 (3)404 -- 包路径是否正确,此包非彼包 (4)404 -- 确认类包是否正确,此类 ...