c++实战篇(二)——基于自旋锁实现的日志服务模块

2024-04-22 06:52

本文主要是介绍c++实战篇(二)——基于自旋锁实现的日志服务模块,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

日志模块一直是服务端开发比较重要的部分,而在实际应用中向日志中进行写入的操作往往不是单线/进程的,而在多进/线程中如何实现对共享资源的保护,就成了一个比较重要的问题,而在实际开发中我们常常会利用锁机制来实现对共享内存的保护,而今天我们所要介绍的就是基于自旋锁实现的日志系统模块。

自旋锁的实现以及选择自旋锁的原因

对于什么是自旋锁,博主在上一篇文章中已经有过介绍,这里就不做赘述了,如果大家有兴趣的话可以去看看我的那篇文章:
C++实战篇(一)——自旋锁的使用
今天由于需求,我对代码进行了重写,代码如下;

class spinlock_mutex  //自旋鎖
{
private:atomic_flag flag;spinlock_mutex(const spinlock_mutex&) = delete;spinlock_mutex& operator=(const spinlock_mutex) = delete;
public:spinlock_mutex() {flag.clear();}void lock()           // 加锁。{while (flag.test_and_set()) //判断是否可以获取到锁;}void unlock()      // 解锁。{flag.clear();}
};

至于为什么我们在日志系统里面会选用自旋锁而不是使用互斥锁,主要我们可以分析一下,在日志模块中,什么是核心需要,是IO操作,而每一次的IO操作的时间其实很短,使用互斥锁需要对阻塞的线/进程频繁的唤醒与阻塞,造成资源的不必要浪费,所以这里我们选择采用自旋锁来加锁比较合适。

日志模块的实现

头文件部分(test.h)

#include "_cmpublic.h"
#include "_public.h"
#include <thread>using namespace std;
using namespace idc;class spinlock_mutex  //自旋鎖
{
private:atomic_flag flag;spinlock_mutex(const spinlock_mutex&) = delete;spinlock_mutex& operator=(const spinlock_mutex) = delete;
public:spinlock_mutex() {flag.clear();}void lock()           // 加锁。{while (flag.test_and_set());}void unlock()      // 解锁。{flag.clear();}
};class clogfile
{
private:ofstream fout; //日志文件的操作类string m_filename; //日志文件名ios::openmode m_mode; //文件的打开方式int m_maxsize; // 单个日志文件的最大大小,超过该值就切换日志bool m_enbuff; //是否启用缓冲bool m_backup; //是否自动切换日志spinlock_mutex m_lock; //自旋锁public:clogfile(int maxsize=100){m_maxsize=maxsize;}bool open(const string& filename,ios::openmode mode=ios::app,const bool enbuff=false,const bool back_up=true);template <typename ...Args>bool write(const char* fmt,Args... args){if (!fout.is_open()){return false;}back_up(); //查看是否要进行换页操作m_lock.lock();fout<<ltime1()<<" "<<sformat(fmt,args...)<<endl; //输入时间以及日志内容m_lock.unlock();return fout.good();}template <typename T>clogfile& operator<<(const T& value){m_lock.lock();fout<<value;m_lock.unlock();return *this;}void close(){fout.close();}~clogfile(){close();}private:bool back_up();
};

main函数及相关函数的实现

#include "test.h"using namespace std;
using namespace idc;clogfile logfile;      // 创建日志对象。void func()
{for (int ii=0;ii<50;ii++){logfile.write("第%d个超女开始表演...",ii);     // 表演前,写一行日志,...表示正在表演中。logfile << "ok.\n";                                          // 表演完成后,写入ok。}
}int main()
{thread t1(func);thread t2(func);// 打开日志文件。if (logfile.open("/tmp/log/demo42.log",ios::app,false)==false){printf("logfile.open(/tmp/log/demo42.log) failed.\n");  return -1;}logfile.write("程序开始运行。\n");t1.join();t2.join();logfile.write("程序运行结束。\n");
}bool clogfile::open(const string& filename,ios::openmode mode,const bool enbuff,const bool back_up)
{if (!fout.is_open()){fout.close();}m_filename=filename;m_mode=mode;m_backup=back_up;m_enbuff=enbuff;newdir(m_filename,true);fout.open(m_filename,m_mode);if(enbuff==false){fout<<unitbuf;}return fout.is_open();
}bool clogfile::back_up()
{if(m_backup==false){return true;}if(!fout.is_open()){return false;}if(fout.tellp()>m_maxsize*1024*1024){m_lock.lock();fout.close();string ffilename=m_filename+ltime1();rename(m_filename.c_str(),ffilename.c_str());fout.open(m_filename,m_mode);if(m_enbuff==false){fout<<unitbuf;}m_lock.unlock();}return true;
}

makefile:

all: testtest: test.cpp g++ -pthread -o test test.cpp _public.cpp -std=c++11clean:rm -f test

补充
这里的命名空间namespace idc以及_public.h_public.cpp是博主自己封装的工具库,不便公开,这里只展示与上述代码相关的部分:

  • newdir函数
 bool newdir(const string &pathfilename, const bool bisfilename){if (pathfilename.empty()){return false;}int pos = 1;while (true){int pos1 = pathfilename.find('/', pos);if (pos1 == string::npos){break;}pos = pos1 + 1;string dir = pathfilename.substr(0, pos1);if (access(dir.c_str(), F_OK) == -1){if (mkdir(dir.c_str(), 0777) == -1){return false;}}}if (bisfilename == false){if (access(pathfilename.c_str(), 0) == -1){if (mkdir(pathfilename.c_str(), 0755) == -1){return false;}}}return true;}bool renamefile(const string &oldfilename, const string &newfilename){if (access(oldfilename.c_str(), 0) == -1)return false;if (newdir(newfilename, true) == false){return false;}if (rename(oldfilename.c_str(), newfilename.c_str()) == -1){return false;}return true;}
  • sformat函数,用于格式化输出
template <typename... Args>string sformat(const char *fmt, Args... args){string str;int len = snprintf(nullptr, 0, fmt, args...); // 得到格式化后字符串的长度。if (len < 0)return str; // 如果调用snprintf失败,返回-1。if (len == 0)return str; // 如果调用snprintf返回0,表示格式化输出的内容为空。;str.resize(len);                          // 为string分配内存。snprintf(&str[0], len + 1, fmt, args...); // linux平台第二个参数是len+1,windows平台是len。return str;}

输出结果:

root@iZuf6ckztbjhtavfplgp0dZ:/tmp/log# cat demo42.log
2024-04-21 13:39:36 程序开始运行。2024-04-21 13:39:36 第0个超女开始表演...
ok.
2024-04-21 13:39:36 第1个超女开始表演...
ok.
2024-04-21 13:39:36 第2个超女开始表演...
ok.
2024-04-21 13:39:36 第3个超女开始表演...
ok.
2024-04-21 13:39:36 第4个超女开始表演...
ok.
2024-04-21 13:39:36 第5个超女开始表演...
ok.
2024-04-21 13:39:36 第6个超女开始表演...
ok.
2024-04-21 13:39:36 第7个超女开始表演...
ok.
2024-04-21 13:39:36 第8个超女开始表演...
ok.
2024-04-21 13:39:36 第9个超女开始表演...
ok.
2024-04-21 13:39:36 第10个超女开始表演...
ok.
2024-04-21 13:39:36 第11个超女开始表演...
ok.
2024-04-21 13:39:36 第12个超女开始表演...
ok.
2024-04-21 13:39:36 第13个超女开始表演...
ok.
2024-04-21 13:39:36 第14个超女开始表演...
ok.
2024-04-21 13:39:36 第15个超女开始表演...
ok.
2024-04-21 13:39:36 第16个超女开始表演...
ok.
2024-04-21 13:39:36 第17个超女开始表演...
ok.
2024-04-21 13:39:36 第18个超女开始表演...
ok.
2024-04-21 13:39:36 第19个超女开始表演...
ok.
2024-04-21 13:39:36 第20个超女开始表演...
ok.
2024-04-21 13:39:36 第21个超女开始表演...
ok.
2024-04-21 13:39:36 第22个超女开始表演...
ok.
2024-04-21 13:39:36 第23个超女开始表演...
ok.
2024-04-21 13:39:36 第24个超女开始表演...
ok.
2024-04-21 13:39:36 第25个超女开始表演...
ok.
2024-04-21 13:39:36 第26个超女开始表演...
ok.
2024-04-21 13:39:36 第27个超女开始表演...
ok.
2024-04-21 13:39:36 第28个超女开始表演...
ok.
2024-04-21 13:39:36 第29个超女开始表演...
ok.
2024-04-21 13:39:36 第30个超女开始表演...
ok.
2024-04-21 13:39:36 第31个超女开始表演...
ok.
2024-04-21 13:39:36 第32个超女开始表演...
ok.
2024-04-21 13:39:36 第33个超女开始表演...
ok.
2024-04-21 13:39:36 第34个超女开始表演...
ok.
2024-04-21 13:39:36 第35个超女开始表演...
ok.
2024-04-21 13:39:36 第36个超女开始表演...
ok.
2024-04-21 13:39:36 第37个超女开始表演...
ok.
2024-04-21 13:39:36 第38个超女开始表演...
ok.
2024-04-21 13:39:36 第39个超女开始表演...
ok.
2024-04-21 13:39:36 第40个超女开始表演...
ok.
2024-04-21 13:39:36 第41个超女开始表演...
ok.
2024-04-21 13:39:36 第42个超女开始表演...
ok.
2024-04-21 13:39:36 第43个超女开始表演...
ok.
2024-04-21 13:39:36 第44个超女开始表演...
ok.
2024-04-21 13:39:36 第45个超女开始表演...
ok.
2024-04-21 13:39:36 第46个超女开始表演...
ok.
2024-04-21 13:39:36 第47个超女开始表演...
ok.
2024-04-21 13:39:36 第48个超女开始表演...
ok.
2024-04-21 13:39:36 第49个超女开始表演...
ok.
2024-04-21 13:39:46 程序运行结束。root@iZuf6ckztbjhtavfplgp0dZ:/tmp/log# cat demo42.log
2024-04-21 13:39:36 程序开始运行。2024-04-21 13:39:36 第0个超女开始表演...
ok.
2024-04-21 13:39:36 第1个超女开始表演...
ok.
2024-04-21 13:39:36 第2个超女开始表演...
ok.
2024-04-21 13:39:36 第3个超女开始表演...
ok.
2024-04-21 13:39:36 第4个超女开始表演...
ok.
2024-04-21 13:39:36 第5个超女开始表演...
ok.
2024-04-21 13:39:36 第6个超女开始表演...
ok.
2024-04-21 13:39:36 第7个超女开始表演...
ok.
2024-04-21 13:39:36 第8个超女开始表演...
ok.
2024-04-21 13:39:36 第9个超女开始表演...
ok.
2024-04-21 13:39:36 第10个超女开始表演...
ok.
2024-04-21 13:39:36 第11个超女开始表演...
ok.
2024-04-21 13:39:36 第12个超女开始表演...
ok.
2024-04-21 13:39:36 第13个超女开始表演...
ok.
2024-04-21 13:39:36 第14个超女开始表演...
ok.
2024-04-21 13:39:36 第15个超女开始表演...
ok.
2024-04-21 13:39:36 第16个超女开始表演...
ok.
2024-04-21 13:39:36 第17个超女开始表演...
ok.
2024-04-21 13:39:36 第18个超女开始表演...
ok.
2024-04-21 13:39:36 第19个超女开始表演...
ok.
2024-04-21 13:39:36 第20个超女开始表演...
ok.
2024-04-21 13:39:36 第21个超女开始表演...
ok.
2024-04-21 13:39:36 第22个超女开始表演...
ok.
2024-04-21 13:39:36 第23个超女开始表演...
ok.
2024-04-21 13:39:36 第24个超女开始表演...
ok.
2024-04-21 13:39:36 第25个超女开始表演...
ok.
2024-04-21 13:39:36 第26个超女开始表演...
ok.
2024-04-21 13:39:36 第27个超女开始表演...
ok.
2024-04-21 13:39:36 第28个超女开始表演...
ok.
2024-04-21 13:39:36 第29个超女开始表演...
ok.
2024-04-21 13:39:36 第30个超女开始表演...
ok.
2024-04-21 13:39:36 第31个超女开始表演...
ok.
2024-04-21 13:39:36 第32个超女开始表演...
ok.
2024-04-21 13:39:36 第33个超女开始表演...
ok.
2024-04-21 13:39:36 第34个超女开始表演...
ok.
2024-04-21 13:39:36 第35个超女开始表演...
ok.
2024-04-21 13:39:36 第36个超女开始表演...
ok.
2024-04-21 13:39:36 第37个超女开始表演...
ok.
2024-04-21 13:39:36 第38个超女开始表演...
ok.
2024-04-21 13:39:36 第39个超女开始表演...
ok.
2024-04-21 13:39:36 第40个超女开始表演...
ok.
2024-04-21 13:39:36 第41个超女开始表演...
ok.
2024-04-21 13:39:36 第42个超女开始表演...
ok.
2024-04-21 13:39:36 第43个超女开始表演...
ok.
2024-04-21 13:39:36 第44个超女开始表演...
ok.
2024-04-21 13:39:36 第45个超女开始表演...
ok.
2024-04-21 13:39:36 第46个超女开始表演...
ok.
2024-04-21 13:39:36 第47个超女开始表演...
ok.
2024-04-21 13:39:36 第48个超女开始表演...
ok.
2024-04-21 13:39:36 第49个超女开始表演...
ok.
2024-04-21 13:39:46 程序运行结束。2024-04-21 13:46:40 程序开始运行。2024-04-21 13:46:40 第0个超女开始表演...
ok.
2024-04-21 13:46:40 第1个超女开始表演...
ok.
2024-04-21 13:46:40 第2个超女开始表演...
ok.
2024-04-21 13:46:40 第3个超女开始表演...
ok.
2024-04-21 13:46:40 第4个超女开始表演...
ok.
2024-04-21 13:46:40 第5个超女开始表演...
ok.
2024-04-21 13:46:40 第6个超女开始表演...
ok.
2024-04-21 13:46:40 第7个超女开始表演...
ok.
2024-04-21 13:46:40 第8个超女开始表演...
ok.
2024-04-21 13:46:40 第9个超女开始表演...
ok.
2024-04-21 13:46:40 第10个超女开始表演...
ok.
2024-04-21 13:46:40 第11个超女开始表演...
ok.
2024-04-21 13:46:40 第12个超女开始表演...
ok.
2024-04-21 13:46:40 第13个超女开始表演...
ok.
2024-04-21 13:46:40 第14个超女开始表演...
ok.
2024-04-21 13:46:40 第15个超女开始表演...
ok.
2024-04-21 13:46:40 第16个超女开始表演...
ok.
2024-04-21 13:46:40 第17个超女开始表演...
ok.
2024-04-21 13:46:40 第18个超女开始表演...
ok.
2024-04-21 13:46:40 第19个超女开始表演...
ok.
2024-04-21 13:46:40 第20个超女开始表演...
ok.
2024-04-21 13:46:40 第21个超女开始表演...
ok.
2024-04-21 13:46:40 第22个超女开始表演...
ok.
2024-04-21 13:46:40 第23个超女开始表演...
ok.
2024-04-21 13:46:40 第24个超女开始表演...
ok.
2024-04-21 13:46:40 第25个超女开始表演...
ok.
2024-04-21 13:46:40 第26个超女开始表演...
ok.
2024-04-21 13:46:40 第27个超女开始表演...
ok.
2024-04-21 13:46:40 第28个超女开始表演...
ok.
2024-04-21 13:46:40 第29个超女开始表演...
ok.
2024-04-21 13:46:40 第30个超女开始表演...
ok.
2024-04-21 13:46:40 第31个超女开始表演...
ok.
2024-04-21 13:46:40 第32个超女开始表演...
ok.
2024-04-21 13:46:40 第33个超女开始表演...
ok.
2024-04-21 13:46:40 第34个超女开始表演...
ok.
2024-04-21 13:46:40 第35个超女开始表演...
ok.
2024-04-21 13:46:40 第36个超女开始表演...
ok.
2024-04-21 13:46:40 第37个超女开始表演...
ok.
2024-04-21 13:46:40 第38个超女开始表演...
ok.
2024-04-21 13:46:40 第39个超女开始表演...
ok.
2024-04-21 13:46:40 第40个超女开始表演...
ok.
2024-04-21 13:46:40 第41个超女开始表演...
ok.
2024-04-21 13:46:40 第42个超女开始表演...
ok.
2024-04-21 13:46:40 第43个超女开始表演...
ok.
2024-04-21 13:46:40 第44个超女开始表演...
ok.
2024-04-21 13:46:40 第45个超女开始表演...
ok.
2024-04-21 13:46:40 第46个超女开始表演...
ok.
2024-04-21 13:46:40 第47个超女开始表演...
ok.
2024-04-21 13:46:40 第48个超女开始表演...
ok.
2024-04-21 13:46:40 第49个超女开始表演...
ok.
2024-04-21 13:46:40 程序运行结束。

结语

今天的有关内容就到此为止啦,有问题的话欢迎在评论区评论,大家可以集思广益,如果你觉得博主的内容对你有帮助,欢迎三连一下和订阅专栏
如果博主文章里面有什么错误页欢迎斧正(毕竟博主页只是个小蒟蒻鸡),好了,大家下篇文章见!

这篇关于c++实战篇(二)——基于自旋锁实现的日志服务模块的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/925078

相关文章

Python正则表达式语法及re模块中的常用函数详解

《Python正则表达式语法及re模块中的常用函数详解》这篇文章主要给大家介绍了关于Python正则表达式语法及re模块中常用函数的相关资料,正则表达式是一种强大的字符串处理工具,可以用于匹配、切分、... 目录概念、作用和步骤语法re模块中的常用函数总结 概念、作用和步骤概念: 本身也是一个字符串,其中

python实现svg图片转换为png和gif

《python实现svg图片转换为png和gif》这篇文章主要为大家详细介绍了python如何实现将svg图片格式转换为png和gif,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录python实现svg图片转换为png和gifpython实现图片格式之间的相互转换延展:基于Py

Python中的getopt模块用法小结

《Python中的getopt模块用法小结》getopt.getopt()函数是Python中用于解析命令行参数的标准库函数,该函数可以从命令行中提取选项和参数,并对它们进行处理,本文详细介绍了Pyt... 目录getopt模块介绍getopt.getopt函数的介绍getopt模块的常用用法getopt模

Python利用ElementTree实现快速解析XML文件

《Python利用ElementTree实现快速解析XML文件》ElementTree是Python标准库的一部分,而且是Python标准库中用于解析和操作XML数据的模块,下面小编就来和大家详细讲讲... 目录一、XML文件解析到底有多重要二、ElementTree快速入门1. 加载XML的两种方式2.

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

Python实现图片分割的多种方法总结

《Python实现图片分割的多种方法总结》图片分割是图像处理中的一个重要任务,它的目标是将图像划分为多个区域或者对象,本文为大家整理了一些常用的分割方法,大家可以根据需求自行选择... 目录1. 基于传统图像处理的分割方法(1) 使用固定阈值分割图片(2) 自适应阈值分割(3) 使用图像边缘检测分割(4)

Android实现在线预览office文档的示例详解

《Android实现在线预览office文档的示例详解》在移动端展示在线Office文档(如Word、Excel、PPT)是一项常见需求,这篇文章为大家重点介绍了两种方案的实现方法,希望对大家有一定的... 目录一、项目概述二、相关技术知识三、实现思路3.1 方案一:WebView + Office Onl

C# foreach 循环中获取索引的实现方式

《C#foreach循环中获取索引的实现方式》:本文主要介绍C#foreach循环中获取索引的实现方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、手动维护索引变量二、LINQ Select + 元组解构三、扩展方法封装索引四、使用 for 循环替代

Spring Security+JWT如何实现前后端分离权限控制

《SpringSecurity+JWT如何实现前后端分离权限控制》本篇将手把手教你用SpringSecurity+JWT搭建一套完整的登录认证与权限控制体系,具有很好的参考价值,希望对大家... 目录Spring Security+JWT实现前后端分离权限控制实战一、为什么要用 JWT?二、JWT 基本结构