C#——单例模式

代码:

复制代码
public class Singleton
{
    private static Singleton _instance;
    private static readonly object locker = new object();

    private Singleton() { }


    public static Singleton GetInstance()
    {
        // 当第一个线程运行到这里时,此时会对locker对象 "加锁",
        // 当第二个线程运行该方法时,首先检测到locker对象为"加锁"状态,该线程就会挂起等待第一个线程解锁
        // lock语句运行完之后(即线程运行完之后)会对该对象"解锁"
        //双重锁定
        if (_instance == null)
        {
            lock (locker)
            {
                // 如果类的实例不存在则创建,否则直接返回
                if (_instance == null)
                {
                    _instance = new Singleton();
                }

            }
        }


        return _instance;
    }



    public void DoBusiness()
    {
        Console.WriteLine("单例对象执行业务逻辑");
    }

    public void ShowInfo(string message)
    {
        Console.WriteLine($"单例对象接收消息:{message}");
    }

}

调用:

复制代码
    internal class SingleTest
    {
        public void DemoMethod()
        {

            Singleton sc = Singleton.GetInstance();

            sc.DoBusiness();
        }


        // 多线程调用示例
        public void MultiThreadDemo()
        {
            // 线程1
            Task.Run(() =>
            {
                Singleton sc = Singleton.GetInstance();
                sc.ShowInfo("线程1的消息");
            });

            // 线程2
            Task.Run(() =>
            {
                Singleton sc = Singleton.GetInstance();
                sc.ShowInfo("线程2的消息");
            });
        }

        //运行后会发现,即使两个线程同时调用,也只会创建一个单例实例。


    }

解释:

复制代码
// 1. 私有静态变量:用于存储Singleton的唯一实例,初始值为null
private static Singleton _instance;

// 2. 私有静态只读锁对象:用于多线程环境下的同步锁,保证线程安全
private static readonly object locker = new object();
  • _instance :这是单例的 "容器",整个程序中只有一个这个静态变量,所以能存储唯一的实例。初始值为null,表示实例还未创建。
  • locker :这是一个 "锁标记",用于多线程时的互斥访问。readonly修饰保证这个锁对象不会被篡改,是线程同步的关键。
复制代码
private Singleton() { }

private Singleton() { }能禁止外部用new创建实例,核心原因是C# 的访问修饰符规则限制了private构造函数的调用范围 ------ 它只能在类的内部被访问,外部代码完全没有权限调用这个构造函数

当用new 类名()创建实例时,C# 编译器会自动调用这个类的构造函数来完成对象的初始化。构造函数是创建实例的 "必经之路",没有调用构造函数,就无法创建类的实例。

复制代码
public static Singleton GetInstance()
{
    // 第一次检查:快速判断实例是否已存在(无锁,提升性能)
    if (_instance == null)
    {
        // 加锁:保证同一时间只有一个线程能进入这个代码块
        lock (locker)
        {
            // 第二次检查:在锁内再次判断实例是否已存在(防止多线程重复创建)
            if (_instance == null)
            {
                // 创建唯一的实例
                _instance = new Singleton();
            }
        }
    }
    // 返回唯一实例
    return _instance;
}

这部分是整个单例模式的核心,也是 "双重检查锁定" 的关键,我们拆成三步理解:

(1)第一次检查(外层if (_instance == null)):提升性能
  • 如果实例已经创建(_instance != null),直接返回实例,不需要加锁。这是性能优化的关键 ------ 因为大部分情况下实例已经存在,加锁是耗时操作,这一步能避免不必要的锁开销。
  • 只有实例还未创建时(_instance == null),才会进入加锁逻辑。
(2)lock (locker):保证线程互斥
  • 当多个线程同时走到这一步时,lock语句会让第一个线程获取locker的锁,其他线程会被挂起等待,直到第一个线程释放锁。
  • 这就保证了同一时间只有一个线程能进入锁内的代码块,避免了多线程同时创建实例的问题。
(3)第二次检查(内层if (_instance == null)):防止重复创建
  • 这是最容易被忽略但最关键的一步!举个例子:
    • 线程 A 和线程 B 同时走到外层if,发现_instance == null
    • 线程 A 先获取锁,进入锁内,此时内层if发现_instance == null,创建实例;
    • 线程 A 释放锁后,线程 B 获取锁,进入锁内 ------ 如果没有内层if,线程 B 会再次创建实例,破坏单例;
    • 有了内层if,线程 B 会发现_instance已经不为null,直接跳过创建步骤,保证实例唯一。
相关推荐
Lv11770084 小时前
Visual Studio 中的字符串
ide·笔记·c#·visual studio
与遨游于天地4 小时前
接口与实现分离:从 SPI 到 OSGi、SOFAArk的模块化演进
开发语言·后端·架构
醇氧4 小时前
Spring Boot 应用启动优化:自定义事件监听与优雅启动管理
java·开发语言·python
请叫我初学者4 小时前
Java学习心得、项目流程(一个Java实习3月的菜鸟)
java·开发语言·intellij-idea·java实习心得
Lv11770084 小时前
Visual Studio中的 var 和 dynamic
ide·笔记·c#·visual studio
代码游侠4 小时前
学习笔记——线程
linux·运维·开发语言·笔记·学习·算法
技术净胜4 小时前
MATLAB基本运算与运算符全解析
开发语言·matlab
企微自动化4 小时前
Java 实现 Token 安全缓存:使用 ReentrantLock 和单例模式实现并发安全的 Token 管理器
开发语言·javascript·ecmascript
古德new4 小时前
openFuyao容器平台:企业级云原生全生命周期管理实践指南
开发语言