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

发布于:2025-06-27 ⋅ 阅读:(21) ⋅ 点赞:(0)

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

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

' 用户WebSocket信息类
    Public Class UserWebSocketInfo
        Public Property SessionID As String
        Public Property WebSocket As WebSocket
        Public Property LastResponseTime As DateTime
        Public Property PendingHeartbeatCount As Integer
        Public Property LockObject As New Object()
    End Class
 Private ClientList As New ConcurrentDictionary(Of String, UserWebSocketInfo)

 Private Sub SendHeartbeatAndCheckTimeouts(state As Object)
        Try
            ' 遍历所有连接用户
            For Each kvp In ClientList
                Dim userInfo = kvp.Value

                SyncLock userInfo.LockObject
                    ' 计算自上次响应以来的时间
                    Dim secondsSinceLastResponse = (DateTime.Now - userInfo.LastResponseTime).TotalMilliseconds '毫秒

                    ' 如果超过超时时间
                    If secondsSinceLastResponse >= RESPONSE_TIMEOUT_Ms Then
                        ' 增加未响应计数
                        userInfo.PendingHeartbeatCount += 1

                        ' 如果重试次数超过最大限制,关闭连接
                        If userInfo.PendingHeartbeatCount > MAX_RETRY_COUNT Then
                            CloseConnection(userInfo.SessionID, WebSocketCloseStatus.NormalClosure, "多次心跳超时")
                            RaiseEvent DataTimeOut(userInfo.SessionID)
                            Debug.Print("用户" & userInfo.SessionID & " 因心跳超时被断开")
                            Continue For
                        End If

                        ' 发送心跳消息
                        SendMessageAsync(userInfo.SessionID, HEARTBEAT_MESSAGE).Wait()
                    End If
                End SyncLock

            Next

        Catch ex As Exception
            debug.print("心跳处理异常: {" & ex.Message)
        End Try
    End 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 在多线程中可能输出混乱,建议使用线程安全的日志组件。

网站公告

今日签到

点亮在社区的每一天
去签到