c++命名对象和匿名对象的解析
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
c++命名对象和匿名对象的解析
最近在看muduo
库,对⾥⾯的⽇志库⽐较感兴趣,于是看到了以下的语句:
刚看到这些语句时,和平时遇到⽇志打印的语句不太⼀样,很疑惑这样⼀条语句是怎么把⽇志打印出来的。
⽹上搜索⼀翻后,学到了⼀个知识点:匿名对象。
其实在平时的编码中我们也经常会遇到匿名对象,只是没有关注。
简单的⼀个匿名对象如:
std::string anonymous = std::string("anonymous");
像按值传递的对象(函数⼊参,函数返回值)都是匿名对象,那匿名对象的特点是什么呢?通过下⾯⼀段代码可知:
1
2
34567891011121314151617181920212223242526272829303132333435363738#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;}
39
40
运⾏结果如下:
1,命名对象(⾮new)在离开作⽤域后,调⽤析构函数。
2,匿名对象在离开定义它的语句后,调⽤析构函数。
了解匿名对象的特点后,回到上⾯的⽇志打印语句,如其中⼀条语句:
#define LOG_TRACE if (CLogger::logLevel() <= CLogger::TRACE) \
CLogger(__FILE__, __LINE__, CLogger::TRACE, __func__).stream()
定义了⼀个匿名对象CLogger ,在调⽤完这条语句后,调⽤其析构函数:
1
234567891011
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 :
1
2345678910111213
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下载。
到此这篇关于c++命名对象和匿名对象的解析的⽂章就介绍到这了,更多相关c++命名对象和匿名对象内容请搜索以前的⽂章或继续浏览下⾯的相关⽂章希望⼤家以后多多⽀持!。