Public ReadOnly LockObj As New Object() '全局多线程锁 VB.NET模块中的LockObj 可以在任意模块或类中使用吧
在 VB.NET 中,模块(Module)中声明的 Public ReadOnly LockObj
可以被其他模块或类访问和使用,但需要注意其可见性范围 和线程安全设计原则:
1. 访问范围:能否跨模块 / 类使用?
-
可以 。模块中的
Public
成员本质上是全局共享的,只要在同一个程序集(Assembly)中,其他模块或类可以直接通过模块名.LockObj
的方式访问(如果在同一个命名空间下,甚至可以省略模块名)。示例:vb
vbnet' 模块1:定义锁对象 Module LockModule Public ReadOnly LockObj As New Object() End Module ' 模块2:使用模块1中的锁对象 Module AnotherModule Public Sub DoSomething() SyncLock LockModule.LockObj ' 跨模块使用 ' 线程安全操作 End SyncLock End Sub End Module ' 类中:使用模块中的锁对象 Public Class MyClass Public Sub DoWork() SyncLock LockModule.LockObj ' 跨类使用 ' 线程安全操作 End SyncLock End Sub End Class
- 注意事项:不建议全局共享一把锁
虽然语法上允许跨模块 / 类使用同一个 LockObj
,但设计上不推荐将一个锁对象用于所有场景,原因如下:
- 锁竞争激烈:所有线程都争夺同一把锁,会导致大量线程阻塞,降低并发性能(比如一个简单的计数操作和一个复杂的数据库操作共用一把锁,会互相拖累)。
- 死锁风险增加:如果多个锁对象交叉使用,全局共享锁可能成为死锁的潜在诱因。
- 职责不清晰:一个锁对象负责保护多个不同的资源,代码维护时难以追溯锁与资源的对应关系。
3. 最佳实践
-
锁对象应与保护的资源绑定 :为每个需要同步的资源(或一组关联资源)单独定义锁对象,且锁对象的访问权限应尽可能严格(
Private
或Friend
),避免被外部随意使用。示例(推荐):vb
vbnet' 模块中:锁对象仅用于保护当前模块的资源 Module ServerStateModule Private _serverInfoState As Integer = 0 ' 私有锁对象,仅本模块可用 Private ReadOnly _stateLock As New Object() ' 用私有锁保护本模块的资源操作 Public Sub AddN(value As Integer) SyncLock _stateLock _serverInfoState += value End SyncLock End Sub End Module
-
如需跨模块共享资源:如果多个模块 / 类确实需要操作同一个共享资源(如全局配置),此时可以共享同一把锁,但需在一个专门的 "资源管理模块" 中定义锁对象和资源,避免散落在多个地方。
总结
- 语法上,模块中的
Public ReadOnly LockObj
可以被其他模块 / 类访问使用。 - 设计上,应尽量避免全局共享一把锁,建议锁对象与被保护资源一一对应,并限制其访问范围(优先
Private
),以提高并发性能和代码可维护性。