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
一、当前代码的锁机制与共享数据访问逻辑
-
数据结构与线程访问场景
ClientList
是ConcurrentDictionary
,本身是线程安全的集合,但其中存储的UserWebSocketInfo
实例内的属性(如LastResponseTime
、PendingHeartbeatCount
)并非线程安全。SendHeartbeatAndCheckTimeouts
方法可能在后台线程中执行(例如定时器触发),而其他线程(如处理 WebSocket 消息的线程)可能同时修改userInfo.LastResponseTime
(当收到客户端响应时)。
-
SyncLock
的作用- 锁
userInfo.LockObject
确保在同一时刻只有一个线程能访问和修改userInfo
的属性(LastResponseTime
、PendingHeartbeatCount
),避免数据竞争(例如同时读取和修改PendingHeartbeatCount
导致计数错误)。
- 锁
二、是否可以去掉锁?需考虑以下情况
(1)如果所有对 UserWebSocketInfo
的操作都在单线程中执行
- 可以不加锁:若
SendHeartbeatAndCheckTimeouts
和所有修改userInfo
的代码(如响应处理)都在同一个线程中运行,不存在多线程竞争,锁可以去掉。 - 实际场景中很少见:WebSocket 服务器通常使用多线程处理不同客户端请求,后台心跳线程和消息处理线程大概率是不同的。
(2)如果存在多线程同时访问 userInfo
- 必须加锁:例如,后台心跳线程正在检查
LastResponseTime
时,消息处理线程同时更新LastResponseTime
,可能导致:- 心跳检查时读取到旧的时间戳,误判超时;
PendingHeartbeatCount
增加和LastResponseTime
更新顺序错乱,导致逻辑错误。
三、ConcurrentDictionary
与内部对象的线程安全区别
ConcurrentDictionary
保证集合本身的操作(添加、删除、查询)是线程安全的,但不保证内部对象的属性访问安全。例如:' 以下操作并非原子性,多线程时可能出错 userInfo.PendingHeartbeatCount += 1 ' 实际是读取-修改-写入三步操作,可能被其他线程中断
- 即使
ClientList
是线程安全集合,内部对象的属性修改仍需额外锁保护。
四、是否有替代方案?
如果想去掉 SyncLock
,可以考虑以下线程安全设计:
- 使用不可变对象:将
UserWebSocketInfo
设计为不可变类,所有属性只读,修改时创建新实例(但需结合ConcurrentDictionary
的TryUpdate
方法实现)。 - 原子操作:对简单数值(如
PendingHeartbeatCount
)使用Interlocked
类进行原子更新,但LastResponseTime
(DateTime
)无法用原子操作处理。 - 读写锁:若读操作多、写操作少,可使用
ReaderWriterLockSlim
优化,但复杂度较高。
五、结论:建议保留锁机制
- 当前代码必须加锁:因为
UserWebSocketInfo
的属性(LastResponseTime
、PendingHeartbeatCount
)在多线程中存在读写竞争,SyncLock
是最简单有效的线程安全方案。 - 例外情况:仅当确认所有对
userInfo
的访问都在单线程中执行时(如纯测试场景),才可去掉锁,否则可能导致超时判断错误、连接异常关闭等问题。
延伸建议
- 优化锁范围:尽量缩小
SyncLock
包裹的代码范围,只锁必要的属性操作,提高并发性能。 - 避免异步中使用
.Wait()
:SendMessageAsync().Wait()
可能导致死锁,建议改为Await
或使用ConfigureAwait(False)
。 - 日志优化:
Debug.Print
在多线程中可能输出混乱,建议使用线程安全的日志组件。