1.如何从零写一个日志库(glog介绍)
2.华为nova10是源码鸿蒙系统吗 - 知百科
3.easylogging源码学习笔记(6)
如何从零写一个日志库(glog介绍)
探索日志管理的艺术,本文将深入解析glog——谷歌开源的分析日志库,为你揭示从零开始构建自定义日志解决方案的源码关键要素。让我们一起从基本需求到高级接口,分析一窥其内部工作原理。源码基本需求与核心组件
glog的分析数字商品源码核心在于LogMessage类,它负责记录日志的源码时间、位置信息以及根据过滤条件进行输出。分析0.3.5版本作为起点,源码提供了诸如LOG、分析LOG_IF和LOG_TO_SINK等接口,源码让你能灵活定制输出到默认目标或自定义sink。分析 获取glog的源码大话随风2.4源码源代码:/google/glog.git,源码中的分析src/glog/http://logging.h.in是你探索之旅的起点。关键接口详解
LOG(severity):这是源码基础接口,将日志发送到默认输出,如LOG_ERROR = LogMessage(FILE, LINE, GLOG_ERROR)。 LOG_IF(severity, condition):条件式过滤,如LOG_IF(severity, !condition) & LOG(severity),只在满足条件时记录。 LOG_TO_SINK(sink, severity):直接将日志导向自定义sink,同时不遗漏默认输出。宏定义的艺术
通过VLOG、LOG_TO_STRING和SYSLOG_IF等宏,你可以根据需求进行灵活的金手指阅读源码抽样和条件输出。例如,VLOG_IF(INFO, condition)在满足条件时增加日志的可见性,VLOG_EVERY_N则进行抽样记录。深入日志流程
glog的过滤机制依赖于LOG_IF,它根据FLAGS_minloglevel调整记录行为,避免不必要的CPU开销。LogMessage的构造和析构处理日志记录,而Stream函数确保与输出操作的兼容性。 在使用时,如LOG(INFO) << "log test",实际上会先检查条件,然后输出,长阴线源码返回值为void以保持与"<<"操作符的配合。高级定制与性能优化
LogMessageData结构体提供了字节的流式输出能力,通过LogDestination和Logger接口,实现灵活的日志目标管理,如支持异步写入的MyAppLogSink。 尽管glog提供了强大的基础,但其在多线程写入和日志文件类型上的限制需要通过sink接口进行扩展,以适应复杂环境下的日志需求。后续章节预告
接下来的文章将深入讨论FATAL日志和coredump的生成过程,带你了解glog在故障诊断中的关键作用。 总之,glog不仅是spring源码编译警告一个强大的日志库,更是日志管理策略的灵活载体。掌握它,为你的项目增添强大的故障排查能力。敬请期待《深入解析glog的FATAL与coredump》。华为nova是鸿蒙系统吗 - 知百科
品牌型号:华为nova
系统:HarmonyOS2
华为nova是鸿蒙系统。HUAWEInova是华为于年7月4日发布的手机;它的操作系统为HarmonyOS2.0。鸿蒙系统是一个开放源代码的分布式操作系统,旨在为各种设备提供统一的操作系统解决方案,具有多种先进的功能和特性,如分布式架构、轻量级化、快速响应等,可以提供更加智能、更加高效的操作系统服务。
HUAWEInova采用6.英寸OLED屏幕;长度为.毫米,宽度为.毫米,厚度为6.毫米,机身重量约克(含电池);提供号色、绮境森林、普罗旺斯、曜金黑四款配色。
HUAWEInova搭载骁龙G4G处理器;后置万像素超感知摄像头+万像素超广角微距摄像头+万人像虚化景深摄像头,前置万像素超广角摄像头;内置毫安时容量电池。
功能特点:
1、影像:HUAWEInova后置万RYYB超感知镜头,搭载华为RYYB超感光滤色阵列,相比传统RGGB提升%进光量;运用后置人像追焦技术,当锁定拍摄目标后,焦点智能跟随主角;搭载人像识别和背景虚化算法,使vlog人像虚化较为自然,并且搭配一键成片等快捷功能。
2、隐私安全:HUAWEInova可以在应用中实现隐私保护,支持在截图聊天界面,一键隐藏除头像昵称外的手机号码、银行卡账户等。
3、游戏体验:HUAWEInova支持TouchTurbo2.0体感触控,轻微晃动机身即可触发相应按键操作;搭载立体声场双扬声器,较宽声场。
4、Wi-Fi6:HUAWEInova搭载增强版Wi-Fi6,支持4KQAM,最高速率提升%,在Wi-Fi信号弱的情况下,也可以实现较低时延。
easylogging源码学习笔记(6)
`LOG` 是默认日志、CLOG自定义日志、LOG_IF条件日志
特殊日志
LOG_EVERY_N、LOG_AFTER_N、LOG_N_TIMES
for (int i = 1; i <= ; ++i) {
LOG_EVERY_N(2, INFO) << "Logged every second iter";
}// 5 logs written; 2, 4, 6, 7,
for (int i = 1; i <= ; ++i) {
LOG_AFTER_N(2, INFO) << "Log after 2 hits; " << i;
}// 8 logs written; 3, 4, 5, 6, 7, 8, 9,
for (int i = 1; i <= ; ++i) {
LOG_N_TIMES(3, INFO) << "Log only 3 times; " << i;
}// 3 logs writter; 1, 2, 3
条件日志和特殊日志可以搭配使用
* `VLOG_IF(condition, verbose-level)`
* `CVLOG_IF(condition, verbose-level, loggerID)`
* `VLOG_EVERY_N(n, verbose-level)`
* `CVLOG_EVERY_N(n, verbose-level, loggerID)`
* `VLOG_AFTER_N(n, verbose-level)`
* `CVLOG_AFTER_N(n, verbose-level, loggerID)`
* `VLOG_N_TIMES(n, verbose-level)`
* `CVLOG_N_TIMES(n, verbose-level, loggerID)`
日志详细等级判定
if (VLOG_IS_ON(2)) {
// Verbosity level 2 is on for this file
}
性能追踪
* `TIMED_FUNC(obj-name)`
* `TIMED_SCOPE(obj-name, block-name)`
* `TIMED_BLOCK(obj-name, block-name)`
这些宏实际上都是关于el::base::type::PerformanceTrackerPtr,一个指向el::base::PerformanceTracker的指针
#if defined(ELPP_FEATURE_ALL) || defined(ELPP_FEATURE_PERFORMANCE_TRACKING)
PerformanceTracker::PerformanceTracker(const std::string& blockName,
base::TimestampUnit timestampUnit,
const std::string& loggerId,
bool scopedLog, Level level) :
m_blockName(blockName), m_timestampUnit(timestampUnit), m_loggerId(loggerId), m_scopedLog(scopedLog),
m_level(level), m_hasChecked(false), m_lastCheckpointId(std::string()), m_enabled(false) {
#if !defined(ELPP_DISABLE_PERFORMANCE_TRACKING) && ELPP_LOGGING_ENABLED
// We store it locally so that if user happen to change configuration by the end of scope
// or before calling checkpoint, we still depend on state of configuration at time of construction
el::Logger* loggerPtr = ELPP->registeredLoggers()->get(loggerId, false);
m_enabled = loggerPtr != nullptr && loggerPtr->m_typedConfigurations->performanceTracking(m_level);
if (m_enabled) {
base::utils::DateTime::gettimeofday(&m_startTime);
}
#endif // !defined(ELPP_DISABLE_PERFORMANCE_TRACKING) && ELPP_LOGGING_ENABLED
}
在构造函数中获取一个时间,
PerformanceTracker::~PerformanceTracker(void) {
#if !defined(ELPP_DISABLE_PERFORMANCE_TRACKING) && ELPP_LOGGING_ENABLED
if (m_enabled) {
base::threading::ScopedLock scopedLock(lock());
if (m_scopedLog) {
base::utils::DateTime::gettimeofday(&m_endTime);
base::type::string_t formattedTime = getFormattedTimeTaken();
PerformanceTrackingData data(PerformanceTrackingData::DataType::Complete);
data.init(this);
data.m_formattedTimeTaken = formattedTime;
PerformanceTrackingCallback* callback = nullptr;
for (const std::pair& h
: ELPP->m_performanceTrackingCallbacks) {
callback = h.second.get();
if (callback != nullptr && callback->enabled()) {
callback->handle(&data);
}
}
}
}
#endif // !defined(ELPP_DISABLE_PERFORMANCE_TRACKING)
}
在析构函数中获取一个时间,处理时间data,使用PerformanceTrackingCallback类型指针callback,并在callback->handle(&data)中处理输出。
由于定义了ELPP_FEATURE_PERFORMANCE_TRACKING,因此在初始化(INITIALIZE_EASYLOGGINGPP)中实际上是安装了一个base::DefaultPerformanceTrackingCallback。
在PerformanceTracker类的handle函数中,callback是一个PerformanceTrackingCallback类型指针,由于安装的是DefaultPerformanceTrackingCallback对象,因此是一个基类指针指向了派生类对象。处理输出的逻辑在DefaultPerformanceTrackingCallback类的handle函数中。
DefaultPerformanceTrackingCallback类的handle函数首先会将数据成员m_data的指针赋值给函数参数,并创建一个base::type::stringstream_t类型的对象ss用于构建输出内容。根据m_data的dataType,输出不同的信息。在输出时,会使用el::base::Writer类构造并输出内容。