关于Qt QReadWriteLock(读写锁) 以及 QSettings 使用的问题

关于Qt QReadWriteLock(读写锁) 使用的问题

经过

我使用这个类实现了一个全局单例对象,用来让程序读取和保存配置文件,但是在软件使用中发生了不明原因的耗时长的问题,经过排查,发现是因为读锁获取时导致的耗时,继而寻找资料结合实际调试经验确认如下问题:

复制代码
当系统进入"写多读少"模式时,QReadWriteLock 的实现通常会优先保障写线程的互斥访问,以避免写饥饿。这导致:
新来的读线程无法获得读锁,即使已有读线程正在运行
写线程持续抢占,形成"写者优先"调度
读线程长时间处于等待队列中,出现"读饥饿"现象
这种机制虽然防止了写操作被无限延迟,但牺牲了读操作的响应性。

引用自QReadLocker在写操作频繁时为何性能下降?

因此读写锁在存在写入操作时,会导致读操作长期处理阻塞转态,且会加剧读队列

QSettings 类

此类是Qt跨平台的配置文件操作类,保证线程及进程安全,并且使用QMap自带数据缓存,理论上直接使用此类是可以满足高速配置文件参数的要求的.

但是经过我的测试,此类在多写的环境中会导致读取性能暴跌,于是翻看源码

查看value 源码

QSettings::value 是调用的QSettingsPrivate::value 函数

其中get函数

复制代码
 const auto locker = qt_scoped_lock(confFile->mutex);

这行代码获取了当前文件对象的排他锁

查看setValue 源码

其中set函数中都是更新的QMap的值

requestUpdate函数,此函数实现了将这个写操作新建成一个事件,并投递到QSettings 实例队列中

然后再查看事件处理函数,里面调用了update函数,update中调用了flush(),flush中调用了sync(),sync中调用了syncConfFile()

syncConfFile函数中实现了将数据写入文件的操作,是个耗时操作具体代码我就不贴了,同志们可以自行查看

sync函数中使用了const auto locker = qt_scoped_lock(confFile->mutex);

即直接把每次的事件都直接使用锁锁住,但是我们的读取操作是需要获取这个锁的

导致的结果就是当写入操作很多的时候,就会导致读取操作阻塞

其实我觉得这个锁操作应该放入syncConfFile函数中,把文件操作那一套操作排除在锁外,就不会有这个问题

完结~

相关推荐
roman_日积跬步-终至千里4 小时前
【Java并发】Java 线程池实战:警惕使用CompletableFuture.supplyAsync
java·开发语言·网络
毕设源码-钟学长4 小时前
【开题答辩全过程】以 基于Springboot的扶贫众筹平台为例,包含答辩的问题和答案
java·spring boot·后端
Nandeska4 小时前
15、基于MySQL的组复制
数据库·mysql
mengzhi啊4 小时前
QUndoView 本质是一个 Qt 界面控件(继承自 QListView),专门适配 QUndoStack
qt
CodeSheep程序羊4 小时前
拼多多春节加班工资曝光,没几个敢给这个数的。
java·c语言·开发语言·c++·python·程序人生·职场和发展
编程小白20264 小时前
从 C++ 基础到效率翻倍:Qt 开发环境搭建与Windows 神级快捷键指南
开发语言·c++·windows·qt·学习
我是咸鱼不闲呀4 小时前
力扣Hot100系列19(Java)——[动态规划]总结(上)(爬楼梯,杨辉三角,打家劫舍,完全平方数,零钱兑换)
java·leetcode·动态规划
深蓝海拓5 小时前
PySide6,QCoreApplication::aboutToQuit与QtQore.qAddPostRoutine:退出前后的清理工作
笔记·python·qt·学习·pyqt
AllData公司负责人5 小时前
AllData数据中台-数据同步平台【Seatunnel-Web】整库同步MySQL同步Doris能力演示
大数据·数据库·mysql·开源
加油,小猿猿5 小时前
Java开发日志-双数据库事务问题
java·开发语言·数据库