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

在分析是否可以去掉 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. 数据结构与线程访问场景

    • ClientListConcurrentDictionary,本身是线程安全的集合,但其中存储的 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 设计为不可变类,所有属性只读,修改时创建新实例(但需结合 ConcurrentDictionaryTryUpdate 方法实现)。
  2. 原子操作 :对简单数值(如 PendingHeartbeatCount)使用 Interlocked 类进行原子更新,但 LastResponseTimeDateTime)无法用原子操作处理。
  3. 读写锁 :若读操作多、写操作少,可使用 ReaderWriterLockSlim 优化,但复杂度较高。

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

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

延伸建议

  1. 优化锁范围 :尽量缩小 SyncLock 包裹的代码范围,只锁必要的属性操作,提高并发性能。
  2. 避免异步中使用 .Wait()SendMessageAsync().Wait() 可能导致死锁,建议改为 Await 或使用 ConfigureAwait(False)
  3. 日志优化Debug.Print 在多线程中可能输出混乱,建议使用线程安全的日志组件。
相关推荐
CodeWithMe1 小时前
【闲谈】对于c++未来的看法
开发语言·c++
忆杰1 小时前
统计用户本月的连续登录天数
开发语言·python
蓝胖子不会敲代码1 小时前
跟着AI学习C# Day26
开发语言·学习·c#
嗜好ya1 小时前
JAVA集合篇--深入理解ConcurrentHashMap图解版
java·开发语言
小猫咪怎么会有坏心思呢2 小时前
华为OD机考-生成哈夫曼树-二叉树(JAVA 2025B卷)
java·开发语言·华为od
卜及中2 小时前
【Python】数据处理工具:Pandas详细指南
开发语言·python·学习·pandas
翱翔的小菜鸟2 小时前
Java Stream API中peek()方法使用不当引发的生产问题
java·开发语言
狐凄3 小时前
Python实例题:基于联邦学习的隐私保护 AI 系统(分布式学习、隐私计算)
开发语言·python
NLxxxxX3 小时前
爬虫获取数据:selenium的应用
开发语言·爬虫·python·selenium·测试工具·numpy·pandas
fs哆哆3 小时前
在VB.net中,文本插入的几个自定义函数
服务器·前端·javascript·html·.net