当前位置: 首页 > news >正文

c#,vb.net LockObject ,多线程锁,多线程安全字典ConcurrentDictionary

在分析是否可以去掉 SyncLock userInfo.LockObject 锁之前,需要先明确多线程环境下的线程安全问题核心:只要存在多个线程同时访问和修改共享数据的可能,就必须考虑线程安全。以下是具体分析:

ClientList 只是用于主线程中获取所有websocket和sessionid,其他线程也不会相互访问,是否可以不加锁?

' 用户WebSocket信息类Public Class UserWebSocketInfoPublic Property SessionID As StringPublic Property WebSocket As WebSocketPublic Property LastResponseTime As DateTimePublic Property PendingHeartbeatCount As IntegerPublic Property LockObject As New Object()End ClassPrivate ClientList As New ConcurrentDictionary(Of String, UserWebSocketInfo)Private Sub SendHeartbeatAndCheckTimeouts(state As Object)Try' 遍历所有连接用户For Each kvp In ClientListDim userInfo = kvp.ValueSyncLock userInfo.LockObject' 计算自上次响应以来的时间Dim secondsSinceLastResponse = (DateTime.Now - userInfo.LastResponseTime).TotalMilliseconds '毫秒' 如果超过超时时间If secondsSinceLastResponse >= RESPONSE_TIMEOUT_Ms Then' 增加未响应计数userInfo.PendingHeartbeatCount += 1' 如果重试次数超过最大限制,关闭连接If userInfo.PendingHeartbeatCount > MAX_RETRY_COUNT ThenCloseConnection(userInfo.SessionID, WebSocketCloseStatus.NormalClosure, "多次心跳超时")RaiseEvent DataTimeOut(userInfo.SessionID)Debug.Print("用户" & userInfo.SessionID & " 因心跳超时被断开")Continue ForEnd If' 发送心跳消息SendMessageAsync(userInfo.SessionID, HEARTBEAT_MESSAGE).Wait()End IfEnd SyncLockNextCatch ex As Exceptiondebug.print("心跳处理异常: {" & ex.Message)End TryEnd Sub

一、当前代码的锁机制与共享数据访问逻辑

  1. 数据结构与线程访问场景

    • ClientList 是 ConcurrentDictionary,本身是线程安全的集合,但其中存储的 UserWebSocketInfo 实例内的属性(如 LastResponseTimePendingHeartbeatCount并非线程安全
    • SendHeartbeatAndCheckTimeouts 方法可能在后台线程中执行(例如定时器触发),而其他线程(如处理 WebSocket 消息的线程)可能同时修改 userInfo.LastResponseTime(当收到客户端响应时)。
  2. SyncLock 的作用

    • 锁 userInfo.LockObject 确保在同一时刻只有一个线程能访问和修改 userInfo 的属性(LastResponseTimePendingHeartbeatCount),避免数据竞争(例如同时读取和修改 PendingHeartbeatCount 导致计数错误)。

二、是否可以去掉锁?需考虑以下情况

(1)如果所有对 UserWebSocketInfo 的操作都在单线程中执行
  • 可以不加锁:若 SendHeartbeatAndCheckTimeouts 和所有修改 userInfo 的代码(如响应处理)都在同一个线程中运行,不存在多线程竞争,锁可以去掉。
  • 实际场景中很少见:WebSocket 服务器通常使用多线程处理不同客户端请求,后台心跳线程和消息处理线程大概率是不同的。
(2)如果存在多线程同时访问 userInfo
  • 必须加锁:例如,后台心跳线程正在检查 LastResponseTime 时,消息处理线程同时更新 LastResponseTime,可能导致:
    • 心跳检查时读取到旧的时间戳,误判超时;
    • PendingHeartbeatCount 增加和 LastResponseTime 更新顺序错乱,导致逻辑错误。

三、ConcurrentDictionary 与内部对象的线程安全区别

  • ConcurrentDictionary 保证集合本身的操作(添加、删除、查询)是线程安全的,但不保证内部对象的属性访问安全。例如:
    ' 以下操作并非原子性,多线程时可能出错
    userInfo.PendingHeartbeatCount += 1  ' 实际是读取-修改-写入三步操作,可能被其他线程中断
    
  • 即使 ClientList 是线程安全集合,内部对象的属性修改仍需额外锁保护。

四、是否有替代方案?

如果想去掉 SyncLock,可以考虑以下线程安全设计:

  1. 使用不可变对象:将 UserWebSocketInfo 设计为不可变类,所有属性只读,修改时创建新实例(但需结合 ConcurrentDictionary 的 TryUpdate 方法实现)。
  2. 原子操作:对简单数值(如 PendingHeartbeatCount)使用 Interlocked 类进行原子更新,但 LastResponseTimeDateTime)无法用原子操作处理。
  3. 读写锁:若读操作多、写操作少,可使用 ReaderWriterLockSlim 优化,但复杂度较高。

五、结论:建议保留锁机制

  • 当前代码必须加锁:因为 UserWebSocketInfo 的属性(LastResponseTimePendingHeartbeatCount)在多线程中存在读写竞争,SyncLock 是最简单有效的线程安全方案。
  • 例外情况:仅当确认所有对 userInfo 的访问都在单线程中执行时(如纯测试场景),才可去掉锁,否则可能导致超时判断错误、连接异常关闭等问题。

延伸建议

  1. 优化锁范围:尽量缩小 SyncLock 包裹的代码范围,只锁必要的属性操作,提高并发性能。
  2. 避免异步中使用 .Wait()SendMessageAsync().Wait() 可能导致死锁,建议改为 Await 或使用 ConfigureAwait(False)
  3. 日志优化Debug.Print 在多线程中可能输出混乱,建议使用线程安全的日志组件。

相关文章:

  • C/C++数据结构之静态数组
  • Matplotlib入门指南:从安装到绘制基本图形
  • 3D制作角色模型的教程-1
  • Java的锁机制问题
  • 【论文阅读笔记】TransparentGS:当高斯溅射学会“看穿”玻璃,如何攻克透明物体重建难题?
  • Protobuf 与 JSON 的兼容性:技术选型的权衡与实践
  • 风险矩阵与灰色综合评价
  • [OS_26] 计算机系统安全 | CIA原则 | 侧信道攻击
  • 【工具】CrossAttOmics:基于交叉注意力的多组学数据整合技术
  • React纯函数和hooks原理
  • 一次性理解Java垃圾回收--简单直接方便面试时使用
  • 华为云Flexus+DeepSeek征文|在Dify-LLM平台中开发童话故事精灵工作流AI Agent
  • java中关于异步转同步的一些解决方案的对比与思考。【spring mvc堵塞式】
  • springboot企业级项目开发之项目测试——集成测试!
  • 【Java】HQL批量增删改
  • 从零理解鱼眼相机的标定与矫正(含 OpenCV 代码与原理讲解)
  • 性能测试之接口关联和函数使用
  • Android14 app被冻结导致进程间通信失败
  • NumPy 数组排序
  • 【Zephyr 系列 28】MCU 闪存文件系统详解:LittleFS + NVS + 块设备设计实战
  • 网站建设明细价单/南京seo排名优化公司
  • html视频网站模板/营销策划经典案例
  • 在哪人网站要以接it项目做/谷歌seo查询
  • 广东制作公司网站/seo新人培训班
  • 网站建设公司推/郑州网站推广方案
  • 谷歌seo外链/无锡seo关键词排名