CEF框架中的一些宏定义(一):CHECK DCHECK

文章目录

  • CHECK & DCHECK
    • DCHECK_IS_ON
  • 日志相关
    • COMPACT_GOOGLE_LOG_FATAL
    • LogMessage
    • LOG_STREAM(FATAL)
    • LAZY_STREAM
    • LogMessageVoidify

在CEF框架中,有很多的宏定义,有必要拿出来说一说,在阅读源码的时候更好的理解。

CHECK & DCHECK

CEF框架中,在很多代码中都用到了CHECK宏和DCHECK宏,这两个宏的用途是一样的,都是对一些变量或者质量做判断,一个用于release版本,一个用于debug版本。

这两个宏定义的主要作用就是用于做某些条件的断言,或者说判断,如果为空或者失败,则记录日志。

下面就一步一步的看:

在这两个宏里面就涉及到了CEF的日志框架,所以在这里也一起简单提一下。

DCHECK(platform_delegate_);
#define CHECK(condition)                       \LAZY_STREAM(LOG_STREAM(FATAL), !(condition)) \<< "Check failed: " #condition ". "#define PCHECK(condition)                       \LAZY_STREAM(PLOG_STREAM(FATAL), !(condition)) \<< "Check failed: " #condition ". "
#define DCHECK(condition)                                         \LAZY_STREAM(LOG_STREAM(DCHECK), DCHECK_IS_ON() && !(condition)) \<< "Check failed: " #condition ". "

DCHECK_IS_ON

用于判断是否打开DCHECK

#if defined(NDEBUG) && !defined(DCHECK_ALWAYS_ON)
#define DCHECK_IS_ON() false
#else
#define DCHECK_IS_ON() true
#endif

在DCHECK和CHECK中的两个宏定义在依次展开。

日志相关

首先,cef框架定义了几个日志等级,就是标准的日志框架中包含的几个等级。

  • 日志等级

    • LOG_VERBOSE

    • LOG_INFO

    • LOG_WARNING

    • LOG_ERROR

    • LOG_FATAL

    • LOG_NUM_SEVERITIES

      typedef int LogSeverity;
      const LogSeverity LOG_VERBOSE = -1;  // This is level 1 verbosity
      // Note: the log severities are used to index into the array of names,
      // see log_severity_names.
      const LogSeverity LOG_INFO = 0;
      const LogSeverity LOG_WARNING = 1;
      const LogSeverity LOG_ERROR = 2;
      const LogSeverity LOG_FATAL = 3;
      const LogSeverity LOG_NUM_SEVERITIES = 4;
      
  • LOG_STREAM宏的定义为:

    // We use the preprocessor's merging operator, "##", so that, e.g.,
    // LOG(INFO) becomes the token COMPACT_GOOGLE_LOG_INFO.  There's some funny
    // subtle difference between ostream member streaming functions (e.g.,
    // ostream::operator<<(int) and ostream non-member streaming functions
    // (e.g., ::operator<<(ostream&, string&): it turns out that it's
    // impossible to stream something like a string directly to an unnamed
    // ostream. We employ a neat hack by calling the stream() member
    // function of LogMessage which seems to avoid the problem.
    #define LOG_STREAM(severity) COMPACT_GOOGLE_LOG_##severity.stream()
    

在LOG_STREAM(FATAL)中,FATAL会替换掉##severity(merging operator),LOG_STREAM(FATAL)展开为COMPACT_GOOGLE_LOG_FATAL。

对应了几个对应的日志宏定义:

#define COMPACT_GOOGLE_LOG_INFO COMPACT_GOOGLE_LOG_EX_INFO(LogMessage)
#define COMPACT_GOOGLE_LOG_WARNING COMPACT_GOOGLE_LOG_EX_WARNING(LogMessage)
#define COMPACT_GOOGLE_LOG_ERROR COMPACT_GOOGLE_LOG_EX_ERROR(LogMessage)
#define COMPACT_GOOGLE_LOG_FATAL COMPACT_GOOGLE_LOG_EX_FATAL(LogMessage)
#define COMPACT_GOOGLE_LOG_DFATAL COMPACT_GOOGLE_LOG_EX_DFATAL(LogMessage)#define COMPACT_GOOGLE_LOG_DCHECK COMPACT_GOOGLE_LOG_FATAL

COMPACT_GOOGLE_LOG_FATAL

COMPACT_GOOGLE_LOG_FATAL对应的宏定义为:

// A few definitions of macros that don't generate much code. These are used
// by LOG() and LOG_IF, etc. Since these are used all over our code, it's
// better to have compact code for these operations.
#define COMPACT_GOOGLE_LOG_EX_INFO(ClassName, ...)                        \::cef::logging::ClassName(__FILE__, __LINE__, ::cef::logging::LOG_INFO, \##__VA_ARGS__)
#define COMPACT_GOOGLE_LOG_EX_WARNING(ClassName, ...)                        \::cef::logging::ClassName(__FILE__, __LINE__, ::cef::logging::LOG_WARNING, \##__VA_ARGS__)
#define COMPACT_GOOGLE_LOG_EX_ERROR(ClassName, ...)                        \::cef::logging::ClassName(__FILE__, __LINE__, ::cef::logging::LOG_ERROR, \##__VA_ARGS__)
#define COMPACT_GOOGLE_LOG_EX_FATAL(ClassName, ...)                        \::cef::logging::ClassName(__FILE__, __LINE__, ::cef::logging::LOG_FATAL, \##__VA_ARGS__)
#define COMPACT_GOOGLE_LOG_EX_DFATAL(ClassName, ...)                        \::cef::logging::ClassName(__FILE__, __LINE__, ::cef::logging::LOG_DFATAL, \##__VA_ARGS__)

LogMessage

ClassName就对应了COMPACT_GOOGLE_LOG_EX_DFATAL(LogMessage)中的LogMessage,这个类就是cef框架中的日志处理类了:

// This class more or less represents a particular log message.  You
// create an instance of LogMessage and then stream stuff to it.
// When you finish streaming to it, ~LogMessage is called and the
// full message gets streamed to the appropriate destination.
//
// You shouldn't actually use LogMessage's constructor to log things,
// though.  You should use the LOG() macro (and variants thereof)
// above.
class LogMessage {public:// Used for LOG(severity).LogMessage(const char* file, int line, LogSeverity severity);// Used for CHECK_EQ(), etc. Takes ownership of the given string.// Implied severity = LOG_FATAL.LogMessage(const char* file, int line, std::string* result);// Used for DCHECK_EQ(), etc. Takes ownership of the given string.LogMessage(const char* file,int line,LogSeverity severity,std::string* result);LogMessage(const LogMessage&) = delete;LogMessage& operator=(const LogMessage&) = delete;~LogMessage();std::ostream& stream() { return stream_; }private:LogSeverity severity_;std::ostringstream stream_;// The file and line information passed in to the constructor.const char* file_;const int line_;#if defined(OS_WIN)// Stores the current value of GetLastError in the constructor and restores// it in the destructor by calling SetLastError.// This is useful since the LogMessage class uses a lot of Win32 calls// that will lose the value of GLE and the code that called the log function// will have lost the thread error value when the log call returns.class SaveLastError {public:SaveLastError();~SaveLastError();unsigned long get_error() const { return last_error_; }protected:unsigned long last_error_;};SaveLastError last_error_;
#endif
};

LOG_STREAM(FATAL)

也就是说,LOG_STREAM(FATAL)的意思就是把内容写到日志文件里。

LAZY_STREAM

在CHECK和DCHECK宏的最外面,还有一个LAZY_STREAM宏定义。

// Helper macro which avoids evaluating the arguments to a stream if
// the condition doesn't hold.
#define LAZY_STREAM(stream, condition) \!(condition) ? (void)0 : ::cef::logging::LogMessageVoidify() & (stream)

stream就是上面提到的日志流,这个宏定义的意思就是如果输入的condition为空(false),则返回0(false);否则则调用LogMessageVoidify & stream

LogMessageVoidify

这个类没什么实际作用,主要作用是抑制宏定义中condition为单独变量时的编译警告。

// This class is used to explicitly ignore values in the conditional
// logging macros.  This avoids compiler warnings like "value computed
// is not used" and "statement has no effect".
class LogMessageVoidify {public:LogMessageVoidify() {}// This has to be an operator with a precedence lower than << but// higher than ?:void operator&(std::ostream&) {}
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/639344.shtml

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

相关文章

PyTorch深度学习实战(31)——生成对抗网络(Generative Adversarial Network, GAN)

PyTorch深度学习实战&#xff08;31&#xff09;——生成对抗网络 0. 前言1. GAN2. GAN 模型分析3. 利用 GAN 模型生成手写数字小结系列链接 0. 前言 生成对抗网络 (Generative Adversarial Networks, GAN) 是一种由两个相互竞争的神经网络组成的深度学习模型&#xff0c;它由…

Flask框架小程序后端分离开发学习笔记《3》客户端向服务器端发送请求

Flask框架小程序后端分离开发学习笔记《3》客户端向服务器端发送请求 Flask是使用python的后端&#xff0c;由于小程序需要后端开发&#xff0c;遂学习一下后端开发。 一、为什么请求数据需要先编码 #构造一个HTTP请求 http_request GET / HTTP/1.1\r\nhost:{}\r\n\r\n.for…

大语言模型系列-ELMo

文章目录 前言一、ELMo的网络结构和流程二、ELMo的创新点总结 前言 在前文大语言模型系列-word2vec已经提到word2vec的缺点&#xff1a; 为每个词汇表中每个分词静态生成一个对应的词向量表示&#xff0c;没有考虑到语境&#xff0c;因此无法无法处理多义词 ps&#xff1a;先…

Python Web 开发之 Flask 入门实践

导语&#xff1a;Flask 是一个轻量级的 Python Web 框架&#xff0c;广受开发者喜爱。本文将带领大家了解 Flask 的基本概念、搭建一个简单的 Web 项目以及如何进一步扩展功能。 一、Flask 简介 Flask 是一个基于 Werkzeug 和 Jinja2 的微型 Web 框架&#xff0c;它的特点是轻…

JAVA RPC Thrift基操实现与微服务间调用

一、Thrift 基操实现 1.1 thrift文件 namespace java com.zn.opit.thrift.helloworldservice HelloWorldService {string sayHello(1:string username) }1.2 执行命令生成Java文件 thrift -r --gen java helloworld.thrift生成代码HelloWorldService接口如下 /*** Autogene…

MBR扇区修复和GRUB引导修复实验

修复MBR扇区 步骤一&#xff1a;在进行实验之前我们需要新加一块磁盘&#xff0c;并对新加磁盘进行分区处理&#xff0c;用来备份sda磁盘的MBR及分区表信息。&#xff08;注&#xff1a;在实验中可以不像我如此这么繁琐&#xff0c;一个主分区&#xff0c;并格式化挂载即可&am…

Android 通过adb命令查看应用流量

一. 获取应用pid号 通过adb shell ps | grep 包名 来获取app的 pid号 二. 查看应用流量情况 使用adb shell cat /proc/#pid#/net/dev 命令 来获取流量数据 备注&#xff1a; Recevice: 表示收包 Transmit: 表示发包 bytes: 表示收发的字节数 packets: 表示收发正确的包量…

【CompletableFuture任务编排】游戏服务器线程模型及其线程之间的交互(以排行榜线程和玩家线程的交互为例子)

需求&#xff1a; 1.我们希望玩家的业务在玩家线程执行&#xff0c;无需回调&#xff0c;因此是多线程处理。 2.匹配线程负责匹配逻辑&#xff0c;是单独一个线程。 3.排行榜线程负责玩家的上榜等。 4.从排行榜线程获取到排行榜列表后&#xff0c;需要给玩家发奖修改玩家数…

【GitHub项目推荐--不错的 C 开源项目】【转载】

大学时接触的第一门语言就是 C语言&#xff0c;虽然距 C语言创立已过了40多年&#xff0c;但其经典性和可移植性任然是当今众多高级语言中不可忽视的&#xff0c;想要学好其他的高级语言&#xff0c;最好是先从掌握 C语言入手。 今天老逛盘点 GitHub 上不错的 C语言 开源项目&…

【代码随想录11】239. 滑动窗口最大值 347. 前 K 个高频元素

目录 239. 滑动窗口最大值题目描述做题思路参考代码 347. 前 K 个高频元素题目描述参考代码 239. 滑动窗口最大值 题目描述 给你一个整数数组 nums&#xff0c;有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧。你只可以看到在滑动窗口内的 k 个数字。滑动窗口每…

详解C语言中`||`的短路机制

在C语言中&#xff0c;逻辑或运算符&#xff08;||&#xff09;是一种常用的逻辑运算符&#xff0c;用于组合多个条件表达式。与其他编程语言一样&#xff0c;C语言中的逻辑或运算符具有短路机制&#xff0c;这是一种非常重要的概念&#xff0c;本文将深入解释C语言中的||短路机…

sportplay项目

1.编写userMapping.xml时报错&#xff0c; Error querying database. Cause: java.sql.SQLSyntaxErrorException: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near ‘‘easyuser’ W…

MSVS C# Matlab的混合编程系列2 - 构建一个复杂(含多个M文件)的动态库:

前言: 本节我们尝试将一个有很多函数和文件的Matlab算法文件集成到C#的项目里面。 本文缩语: MT = Matlab 问题提出: 1 我们有一个比较复杂的Matlab文件: 这个MATLAB的算法,写了很多的算法函数在其他的M文件里面,这样,前面博客的方法就不够用了。会报错: 解决办法如下…

华为机考入门python3--(0)模拟题2-vowel元音字母翻译

分类&#xff1a;字符串 知识点&#xff1a; 字符串转list&#xff0c;每个字符成为list中的一个元素 list(string) 字符串变大小写 str.upper(), str.lower() 题目来自【华为招聘模拟考试】 # If you need to import additional packages or classes, please import …

分享5款简单实用的软件,值得收藏

​ 电脑上的各类软件有很多&#xff0c;除了那些常见的大众化软件&#xff0c;还有很多不为人知的小众软件&#xff0c;专注于实用功能&#xff0c;简洁干净、功能强悍。 1.自定义图标——TileIconifier ​ TileIconifier 是一款可以自定义 Windows 开始菜单图标的软件&#…

蓝牙运动耳机什么牌子的好?2024年运动无线耳机推荐

​在选择运动耳机时&#xff0c;我们需要综合考虑音质、舒适度以及适应不同运动场景的能力。好的运动耳机能够提高运动效率&#xff0c;增添锻炼的乐趣。今天&#xff0c;我为大家介绍几款在音质、佩戴舒适度、防水防汗等方面表现卓越的运动耳机&#xff0c;助你选购最适合的一…

《GreenPlum系列》GreenPlum初级教程-05GreenPlum语言DDLDMLDQL

文章目录 第五章 DDL&DML&DQL1.DDL(Data Definition Language)数据定义语言1.1 创建数据库1.2 查询数据库1.3 删除数据库1.4 创建表1.5 修改表1.6 清除表1.7 删除表 2.DML(Data Manipulation Language)数据操作语言2.1 数据导入2.2 数据更新和删除2.3 数据导出 3.DQL(D…

04 单链表

目录 链表的概念和结构单链表OJ练习 1. 链表的概念和结构 1.1 链表的概念 链表是一种物理存储结构上非连续、非顺序的存储结构&#xff0c;数据元素的逻辑顺序是通过链表中的指针链接次序实现的 1.从上图可以看出链式结构在逻辑上是连续的&#xff0c;物理上不一定连续 2.现…

139 删除链表中的重复元素II

问题描述&#xff1a;存在一个按照升序排列的链表&#xff0c;给你这个链表的头结点head&#xff0c;请你删除链表中所有存在数字重复情况的节点&#xff0c;只保留链表中没有出现的数字&#xff0c;返回的结果同样按升序的结果链表。 求解思路&#xff1a;双指针求解&#xf…

常用的gpt-4 prompt words收集4

1. it poses a certain risk to my work 这对我来说是一个风险点 2. one point to note is that 需要说的一个问题是 3. What is the English phonetic transcription of ‘emoji’? emoji的音标是什么&#xff1f; 4. it would be best if you can insert some proper …