应避免手动实现双重检查锁单例,.NET 4.0+ 优先用 Lazy(需静态字段+异常处理),旧框架用静态构造函数;Lazy 不支持异步,需异步初始化时改用 AsyncLazy 或 Task 封装;慎用全局单例,优先考虑 DI 的 scoped 生命周期。为什么不用 lock + 双重检查就别碰单例手动用 lock 和 if (instance == null) 套两层判断,看似稳妥,实则极易出错:忘记加 volatile 修饰字段会导致指令重排,.NET 4.0 之前还可能因 JIT 优化让未初始化对象被其他线程看到。这不是理论风险------真实项目里出现过构造函数没跑完、instance 却已非 null 的诡异 crash。实操建议:- 绝对不要自己手写双重检查锁(Double-Checked Locking)- .NET 4.0+ 直接用 Lazy<t></t>,它底层已处理内存屏障和线程同步- 若必须兼容旧框架(如 .NET 3.5),改用静态构造函数方式,靠 CLR 保证只执行一次且线程安全Lazy 初始化失败时怎么捕获异常Lazy<t></t> 默认是"第一次访问 Value 时才执行工厂函数",但如果工厂函数抛异常,这个异常会被缓存------后续每次读 Value 都直接 rethrow,不是重新尝试初始化。很多人误以为会重试,结果日志里反复看到同一个 NullReferenceException 却查不到源头。实操建议:- 显式传入 LazyThreadSafetyMode.ExecutionAndPublication(默认值,但写出来更清晰)- 工厂函数内必须包住所有可能异常,或至少 log 出错上下文- 检查是否真需要延迟加载:如果构造开销不大,直接用静态只读字段反而更透明- 示例:private static readonly Lazy<MyService> _instance = new Lazy<MyService>(() => { try { return new MyService(); } catch (Exception ex) { Log.Error(ex); throw; } }, LazyThreadSafetyMode.ExecutionAndPublication);Singleton 实例被 GC 回收?那是你用了错误的 Lazy 构造方式如果用的是 new Lazy<t>(Func<t>)</t></t> 无参构造,实例生命周期由 GC 管理,一旦没有强引用指向 Lazy<t></t> 对象本身,整个延迟初始化机制就失效了------下次再 new 一个 Lazy<t></t>,又会重新创建实例。这不是单例,是"伪单例"。 arXiv Xplorer ArXiv 语义搜索引擎,帮您快速轻松的查找,保存和下载arXiv文章。
相关推荐
huaiixinsi1 小时前
Canal + Outbox、Kafka 选型与高可用、Caffeine 底层原理总结X56611 小时前
什么是Bootstrap的移动优先响应式设计m0_470857641 小时前
实现一个可精确定位、支持左右移动与删除的文本光标系统m0_591364731 小时前
mysql如何通过索引减少行锁范围_mysql索引与加锁逻辑川冰ICE1 小时前
Python爬虫实战⑲|Pandas数据合并与重塑,多数据源整合代码中介商1 小时前
MySQL 核心进阶:事务、隔离级别与视图实战七爷不在我这里1 小时前
oracle的26版本及以下 Null的判断及空串判定acanab1 小时前
isaaclab资产打包的一种方式Be reborn1 小时前
从一行 CSV 到一次浏览器操作:关键字驱动执行引擎设计