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,直接跳过创建步骤,保证实例唯一。
相关推荐
qq_401700412 小时前
QT C++ 好看的连击动画组件
开发语言·c++·qt
t198751282 小时前
广义预测控制(GPC)实现滞后系统控制 - MATLAB程序
开发语言·matlab
报错小能手3 小时前
线程池学习(六)实现工作窃取线程池(WorkStealingThreadPool)
开发语言·学习
一条咸鱼_SaltyFish3 小时前
[Day10] contract-management初期开发避坑指南:合同模块 DDD 架构规划的教训与调整
开发语言·经验分享·微服务·架构·bug·开源软件·ai编程
额呃呃3 小时前
STL内存分配器
开发语言·c++
七点半7703 小时前
c++基本内容
开发语言·c++·算法
嵌入式进阶行者3 小时前
【算法】基于滑动窗口的区间问题求解算法与实例:华为OD机考双机位A卷 - 最长的顺子
开发语言·c++·算法
No0d1es3 小时前
2025年12月 GESP CCF编程能力等级认证Python三级真题
开发语言·php
lalala_lulu3 小时前
什么是事务,事务有什么特性?
java·开发语言·数据库
CCPC不拿奖不改名3 小时前
python基础:python语言中的函数与模块+面试习题
开发语言·python·面试·职场和发展·蓝桥杯