程序设计:C++11原子 写优先的读写锁(源码详解二:操作跟踪)

本文承接程序设计:C++11原子 写优先的读写锁(源码详解)-CSDN博客

上文已经列出了完整代码,完整代码里面增加了操作跟踪,这里就讲解一下这部分是如何实现的。

操作跟踪有两个层面:进程层面和线程层面。

由于这个类设计为一个实体资源,不可移动,不可复制,多线程操作同一个对象实例,因此要跟踪就要在内部区别每个线程,所以用了线程存储对象。进程级的跟踪当然比较简单,用变量记录操作次数即可。

线程存储对象thread_local也是C++11新增的线程库的功能,只需要这么一个存储指示就可以把变量分线程存储。实际实现类似于一个线程相关的静态变量。

进程级跟踪:

cpp 复制代码
		//进程操作计数,防止操作顺序错误
		mutable atomic<int> count_WLock{ 0 };
		mutable atomic<int> count_RLock{ 0 };

线程级跟踪:

cpp 复制代码
		//线程操作记录,防止线程操作错误并可用于中途让出再重新锁定
		struct thread_data
		{
			bool _isLocked{ false };//是否已经锁定,若已经锁定则不重复锁定
			bool _isWLock{ false };//是否是写锁定,当isLocked时有效

            。。。。。。
    	};
	public:
		thread_data* getThreadData()const
		{
			thread_local map<CZS_RWMutex2 const*, thread_data > d;//通过对象地址区分不同的对象
			return &d[this];
		}

getThreadData()获取线程存储对象。因为这个变量相当于一个线程的静态变量,而我们可能会有多个互斥对象要操作,从一个静态变量入口如何区分不同对象?这就需要做成一个容器。

每个操作之后都会同时修改进程操作计数和线程操作计数:

cpp 复制代码
		void after_WLock()const
		{
			++count_WLock;
			getThreadData()->thread_data_WLock();
		}
		void after_RLock()const
		{
			++count_RLock;
			getThreadData()->thread_data_RLock();
		}
		void after_WUnLock()const
		{
			--count_WLock;
			getThreadData()->thread_data_UnLock();
		}
		void after_RUnLock()const
		{
			--count_RLock;
			getThreadData()->thread_data_UnLock();
		}

这样就可以在怀疑锁定操作出问题的时候输出状态来检查了。

编程有时候就像是个游戏。

(这里是结束)

相关推荐
代码中介商6 天前
Linux多线程编程完全指南(下):线程同步与互斥锁
linux·redis·线程·互斥锁
小此方8 天前
Re:思考·重建·记录 现代C++ C++11篇 (四)C++ Lambda 全解析:编译器是如何为你生成仿函数的?
开发语言·c++·c++11·现代c++
结衣结衣.10 天前
手把手教你实现文档搜索引擎
linux·c++·搜索引擎·开源·c++11
十五年专注C++开发12 天前
C++中TAS和CAS实现自旋锁
c++·cas·原子操作·tas
2401_8920709813 天前
【C++11 后端实战】FixedThreadPool 固定线程池完整详解
c++11·生产者消费者·固定线程池
老四啊laosi16 天前
[C++进阶] 25. C++11新特性(一)
c++·c++11·右值
量子炒饭大师17 天前
【C++11】RAII 义体加装指南 ——【包装器 与 异常】C++11中什么是包装器?有哪些包装器?C++常见异常有哪些?(附带完整代码讲解)
开发语言·c++·c++11·异常·包装器
xiaoye-duck18 天前
【C++:C++11】C++11新特性深度解析:从类新功能、Lambda表达式到包装器实战
开发语言·c++·c++11
H Journey18 天前
C++11 新特性 万能函数容器之std::function
c++11·function·万能函数容器
xiaoye-duck21 天前
【C++:C++11】核心进阶:C++11引用折叠、完美转发与可变参数模板实战详解
开发语言·c++·c++11