为什么直接 new Singleton() 不行
单例的核心是“全局唯一实例”,但裸写
private static Singleton instance = new Singleton();会触发类静态字段初始化,而 C# 的静态构造函数和字段初始化顺序在多线程下不保证原子性——尤其在 .NET Framework 早期版本中,可能创建多个实例。更关键的是,它无法控制初始化时机(比如依赖配置加载后才该创建),也不支持延迟加载。
推荐用 Lazy 实现线程安全单例
Lazy<t></t>是 .NET 4.0+ 内置的线程安全延迟初始化类型,默认使用
LazyThreadSafetyMode.ExecutionAndPublication,能确保只执行一次工厂逻辑、且所有线程看到的都是同一个实例。 无需手动加锁,无死锁风险 初始化失败会缓存异常,后续调用直接抛出,行为可预测 支持传入自定义工厂函数,便于注入依赖或做条件判断
public sealed class Singleton
{
private static readonly Lazy<Singleton> _instance = new Lazy<Singleton>(() => new Singleton());
<pre class='brush:php;toolbar:false;'>public static Singleton Instance => _instance.Value;
private Singleton() { } // 私有构造,防止外部 new}
双重检查锁定(DCL)还能用吗
能用,但必须严格满足三个条件:字段用
volatile、两次判空、锁内再次判空。稍有遗漏就会在 x86/x64 指令重排下失效,导致部分线程拿到未完全构造的对象(表现为字段为默认值或 NullReferenceException)。
volatile防止编译器/CPU 重排构造函数指令 第一次判空避免无谓加锁;第二次判空防止多个线程同时通过第一层检查后重复初始化 .NET Core 2.1+ 对
volatile语义更严格,但 DCL 仍比
Lazy<t></t>多出几条 IL 指令和锁开销
public sealed class Singleton
{
private static volatile Singleton _instance;
private static readonly object _lock = new object();
<pre class='brush:php;toolbar:false;'>public static Singleton Instance
{
get
{
if (_instance == null)
{
lock (_lock)
{
if (_instance == null)
_instance = new Singleton();
}
}
return _instance;
}
}
private Singleton() { }}
静态构造函数方式的隐含限制
写成
static Singleton() { _instance = new Singleton(); } 看似简洁,但它会在**首次访问任意静态成员或创建实例时触发**,时机不可控;而且一旦抛出异常,该类型将永久不可用(TypeInitializationException),连
Instance属性都无法再访问。 无法捕获初始化异常并降级处理 不支持参数化构造(比如读取 appsettings.json 后再创建) 在 ASP.NET Core 中,若单例依赖
IConfiguration,静态构造函数根本拿不到服务提供者
真正需要“绝对首次访问即创建”的场景极少,多数时候反而要避开它。
