c++命名对象和匿名对象 c++命名对象和匿名对象的解析
原十 人气:2想了解c++命名对象和匿名对象的解析的相关内容吗,原十在本文为您仔细讲解c++命名对象和匿名对象的相关知识和一些Code实例,欢迎阅读和指正,我们先划重点:c++命名对象,c++匿名对象,下面大家一起来学习吧。
最近在看muduo库,对里面的日志库比较感兴趣,于是看到了以下的语句:
刚看到这些语句时,和平时遇到日志打印的语句不太一样,很疑惑这样一条语句是怎么把日志打印出来的。网上搜索一翻后,学到了一个知识点:匿名对象。其实在平时的编码中我们也经常会遇到匿名对象,只是没有关注。简单的一个匿名对象如:
std::string anonymous = std::string("anonymous");
像按值传递的对象(函数入参,函数返回值)都是匿名对象,那匿名对象的特点是什么呢?通过下面一段代码可知:
#ifndef __CLOGER_H__ #define __CLOGER_H__ #include <string> #include <stdlib.h> #include <stdio.h> class CLoger { public: explicit CLoger(std::string &str): mStr(str) { } ~CLoger() { printf("destructor mStr = %s\n", mStr.c_str()); } std::string &string() { return mStr; } private: std::string mStr; }; #endif #include "anonymous_object.h" int main() { std::string name("name"); CLoger loger(name); //具名对象,main函数退出后才会销毁 std::string anonymous("anonymous"); CLoger(anonymous).string(); //匿名对象,使用完即销毁,即此语句结束后立即调用其析构函数 printf("main finish!\n"); return 0; }
运行结果如下:
1,命名对象(非new)在离开作用域后,调用析构函数。
2,匿名对象在离开定义它的语句后,调用析构函数。
了解匿名对象的特点后,回到上面的日志打印语句,如其中一条语句:
#define LOG_TRACE if (CLogger::logLevel() <= CLogger::TRACE) \
CLogger(__FILE__, __LINE__, CLogger::TRACE, __func__).stream()
定义了一个匿名对象CLogger,在调用完这条语句后,调用其析构函数:
CLogger::~CLogger() { mImpl.finish(); //打印结尾添加文件名和行号 const CLogStream::Buffer& buf(stream().buffer()); g_output(buf.data(), buf.length()); //函数指针调用,默认是输出到标准输出stdout,这里是日志最终输出的语句 if (mImpl.mLevel == FATAL) { g_flush(); abort(); } }
所以,只要调用语句如LOG_INFO:
int main(int argc, char* argv[]) { char name[256]; strncpy(name, argv[0], 256); CAsyncLogging log(::basename(name), kRollSize); log.start(); g_asyncLog = &log; bool longLog = argc > 1; // bench(longLog); LOG_INFO << "loggingTest!"; return 0; }
这个日志库已经被我抽离出来,可以单独编译一个日志库,有兴趣的同学可以到git下载。
加载全部内容