c++ map 析构函数_基于析构函数,实现流方式输出日志的功能

学习面向对象(如C++编程语言),那么肯定了解析构函数,它在对象销毁的时候被调用,通常情况下,构造函数中申请资源,相对应的在析构函数中释放资源。那么析构函数在实现以流方式输出日志中有什么妙用呢?接下来请让我一步步为你揭开这层迷雾。

092b6600a0f3c541d7b742e46fa475e2.png

C/C++语言日志输出模式一般有两种,一种类似printf的方式,另一种类似std::cout的方式,这里说的流方式输出日志指的就是类似std::cout的方式,并且自定义日志输出的格式,同时既可以将日志输出到终端,也可以将日志输出到文件。

855960f8bf5deebaa812fa32a3619680.png

一、格式化字符串的输出流

C++语言提供了ostringstream模版,它支持格式化字符串输出流。

首先让我们看看ostringstream的简单使用,定义ostringstream变量oss,然后将当前的线程id以十六进制的方式写入ostringstream变量, 再调用ostringstream的函数str(),将其转换为std::string字符串之后,打印输出到终端。

bdb3e9de592f089f76e41714aad3adfa.png

输出的信息如下所示,当前的线程id是以十六进制的格式输出。

79da862d16d58b34ae96225654c513ce.png

上面是ostringstream的简单使用方法,那么下面来说明如何构造输出函数名称和行号的字符串。通过利用系统提供的宏定义__func__和__LINE__来构造所需字符串信息。

a1635fcc5d2548fb0e57e099e9c9b4b2.png

从输出的格式内容看,ostringstream按照预期的效果输出了正确的字符串格式。

042543d98d333673efff66c2e03a0ade.png

二、资源获取即初始化

RAII全称是“Resource Acquisition is Initialization”,资源获取即初始化”,简单来说,就是说在构造函数中申请分配资源,在析构函数中释放资源,比如,构造函数中通过new申请内存,析构函数中通过delete释放内存。

基于RAII的思想,我们实现资源管理的管理类,管理类ResourceManager构造函数接受std::function类型的变量, 将其赋值给类的私有成员exit_handle,析构函数内调用exit_handle, 那么如果想要实现满足RAII, 那么只要构建释放资源的std::function类型的变量,然后传递给 ResourceManager。

6e5705489c3ec5b7a4e4a082beaeaf27.png

申请创建内存,然后再创建ResourceManager对象,构造函数的入参是一个匿名函数,函数的功能是释放创建的内存。

0147b376a007fa942ad36485374aa29a.png

运行程序之后,输出打印信息如下所示,说明申请的内存被成功释放。

1fab2c5b4caef923fb8a1ac8e43d7ffd.png

同样的方式,我们可以创建文件之后,再创建ResourceManager对象,构造函数的参数功能是释放文件句柄。

db75c63ea2f4f491e03de1d3e62ad03f.png

运行程序之后,从输出打印信息看,文件句柄使用完成之后,也成功关闭了。

5342c8acc160a4917be62d82b5f0d292.png

从上面的两个例子中,可以看出它们都是利用对象在销毁时会调用析构函数的原理来实现,简单来说,申请资源之后,紧接着设置释放资源,等到申请的资源使用完成之后,资源管理对象在退出作用域之后,就会调用析构函数来释放资源,这样做的好处是,我们不必关注资源什么时候进行释放的问题,同时一定程度上也防止忘记释放资源的问题。

三、利用析构函数来实现日志输出

结合std::ostringstream可以格式化输出流的功能和对象销毁时调用析构函数的原理,我们就可以实现自定义格式,并以流方式输出日志的功能。

实现JWriter类来格式化日志信息并输出,这里我们只是简单输出到终端,当然,你也可以将自定义格式的日志信息写入文件或者写入队列,再由线程将队列中的日志信息写入文件。JWriter类的构造函数接受三个参数:日志等级、函数名称、行号;并且重载了operator<

33fe0909986cd06453d085caeedbd04b.png
4143ed62c88746e8f751b0b59f6943b0.png

那么如何来使用JWriter类,使用效果又是怎样呢?其实很简单,定义如下所示的宏,该宏只接受日志等级的字符串参数。

efebb3b8d20593fcbfa43ade15d5b213.png

调用方式如下所示,它跟我们熟悉使用的std::cout的方式是一样一样的,只是std::cout换成了我们实现的MyLogJ()宏,因此,不存在需要花费时间来学习使用方法的问题。

2159712f77eff0fdc9a31f712d66e053.png

如下所示输出的效果,它首先输出日期时间,然后是函数名和对应行号以及日志等级,最后才输出用户输入的日志信息。这样的格式,通常是比较美观,并且利于问题的定位,当然,你也可以根据个人的喜好来修改JWriter的构造函数来自定义自己的日志格式。

2f29f269eab39ca5db14ba20b61c8f70.png

四、总结

自定义日志格式并以流方式输出的功能已经介绍结束,它是利用了std::ostringstream可以格式化输出流的功能,并且在构造函数格式化日志信息,析构函数最后处理输出日志信息,同时重载了operator<

析构函数不只是用于释放资源,我们可以利用它的特性来做其他的运用,就如本文介绍的一样,利用了析构函数实现了流方式的日志功能,当然,析构函数可能还有其他妙用,这需要我们不断去发掘、创造。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值