c# 设计模式之单例模式

 

单例模式:一个类在内存中只有一个对象(实例),并且提供一个可以全局访问或者获取这个对象的方法。

这两天学的,写了个小例子,问了同事一些关于线程的问题,还有从网上查了一些资料。还犯了一些低级的错误。

vs2017控制台输出文字乱码,从网上找了一些方法不管用,最后发现是自己新建项目选错模板了,选择了.NET CORE的模板,所以才会输出乱码,大家一定要吸取教训。

直接上代码

演示类,Person.cs

复制代码
    public class Person
    {
        /// <summary>
        /// 实例化一个私有静态变量,存储类本身的实例
        /// </summary>
        private static Person _person  = null;

        /// <summary>
        /// 构造函数
        /// </summary>
        private Person()
        {
            Console.WriteLine("构造了一个{0}",GetType().Name);
        }
      
        public static Person GetInstance()
        {
            if(_person == null)
                _person = new Person();
            return _person;
        }    
    }
复制代码

客户端代码:

{
  var person1 = Person.GetInstance();
  var person2 = Person.GetInstance();
  var person3 = Person.GetInstance();
Console.WriteLine("person1 == person2:{0}", object.ReferenceEquals(person1, person2)); }

输出结果:

只输出了一次,两个对象引用相等。说明单例模式没问题。

 

 进阶:

复制代码
 public class Person
 {
        /// <summary>
        /// 实例化一个私有静态变量,存储类本身的实例
        /// </summary>
        private static Person _person  = null;

        /// <summary>
        /// 构造函数
        /// </summary>
        private Person()
        {
            Console.WriteLine("构造了一个{0}",GetType().Name);
        }

        public static Person GetInstance()
        {
            if (_person == null)
                _person = new Person();
            return _person;
        }
}
复制代码

客户端调用代码:

复制代码
           {
                Person person1 = null;
                Person person2 = null;
                Person person3 = null;
                //多线程下可以输出多次
                var thread1 = new Thread(() => { person1 = Person.GetInstance(); });
                var thread2 = new Thread(() => { person2 = Person.GetInstance(); });
                var thread3 = new Thread(() => { person3 = Person.GetInstance(); });
                thread1.Start();
                thread2.Start();
                thread3.Start();
                Thread.Sleep(1000);//等待子线程完成
                Console.WriteLine("person1 == person2:{0}", object.ReferenceEquals(person1, person2));
            }
复制代码

输出结果:

输出了多次,引用也不相等。说明多次实例化这个类,单例模式写的不完全正确,那让我们加上线程安全验证。

 

继续进阶: 

复制代码
 public class Person
 {
        /// <summary>
        /// 实例化一个私有静态变量,存储类本身的实例
        /// </summary>
        private static Person _person  = null;

        /// <summary>
        /// 作为锁的对象,使用私有的、静态的并且是只读的对象
        /// </summary>
        private static readonly object _obj = new object();

        /// <summary>
        /// 构造函数
        /// </summary>
        private Person()
        {
            Console.WriteLine("构造了一个{0}",GetType().Name);
        }

        /// <summary>
        /// 获取类唯一的实例对象
        /// </summary>
        public static Person GetInstance()
        {
            if (_person == null)//先判断是否为空
            {
                lock (_obj)//再判断下是否有别的线程在使用
                {
                    if (_person == null)//等其他线程使用完成后再判断是否为空
                    {
                        _person = new Person();
                    }
                }
            }
            return _person;
        }
  }
复制代码

客户端调用代码:

复制代码
            {
                //使用锁,锁住的对象:使用私有的、静态的并且是只读的对象
                Person person1 = null;
                Person person2 = null;
                Person person3 = null;
                //多线程下可以输出多次
                var thread1 = new Thread(() => { person1 = Person.GetInstance(); });
                var thread2 = new Thread(() => { person2 = Person.GetInstance(); });
                var thread3 = new Thread(() => { person3 = Person.GetInstance(); });
                thread1.Start();
                thread2.Start();
                thread3.Start();
                Thread.Sleep(1000);//等待子线程完成
                Console.WriteLine("person1 == person2:{0}", object.ReferenceEquals(person1, person2));
            }
复制代码

输出结果:

输出一次,引用相等,说明单例模式成功,线程安全已经加上。

进阶2

可以使用静态构造函数作为单例模式:

复制代码
    public class Person
    {
        /// <summary>
        /// 实例化一个私有静态变量,存储类本身的实例
        /// </summary>
        private static Person _person  = null;

        /// <summary>
        /// 构造函数
        /// </summary>
        private Person()
        {
            Console.WriteLine("构造了一个{0}",GetType().Name);
        }
        
        /// <summary>
        /// 静态构造函数,只执行一次
        /// </summary>
        static Person()
        {
            _person = new Person();
        }
        
        /// <summary>
        /// 获取类的实例
        /// </summary>
        public static Person GetInstance()
        {
            return _person;
        }
    }
复制代码

客户端代码:

复制代码
         {
                //使用锁,锁住的对象:使用私有的、静态的并且是只读的对象
                //使用静态构造函数,在里面初始化person对象
                Person person1 = null;
                Person person2 = null;
                Person person3 = null;
                //多线程下可以输出多次
                var thread1 = new Thread(() => { person1 = Person.GetInstance(); });
                var thread2 = new Thread(() => { person2 = Person.GetInstance(); });
                var thread3 = new Thread(() => { person3 = Person.GetInstance(); });
                thread1.Start();
                thread2.Start();
                thread3.Start();
                Thread.Sleep(1000);//等待子线程完成
                Console.WriteLine("person1 == person2:{0}", object.ReferenceEquals(person1, person2));
            }
复制代码

输出结果:

 

 输出一次,引用相等,静态构造函数也可以作为单例模式实现的一种方法。

 

posted @   大稳·杨  阅读(734)  评论(0编辑  收藏  举报
编辑推荐:
· 一个奇形怪状的面试题:Bean中的CHM要不要加volatile?
· [.NET]调用本地 Deepseek 模型
· 一个费力不讨好的项目,让我损失了近一半的绩效!
· .NET Core 托管堆内存泄露/CPU异常的常见思路
· PostgreSQL 和 SQL Server 在统计信息维护中的关键差异
阅读排行:
· DeepSeek “源神”启动!「GitHub 热点速览」
· 微软正式发布.NET 10 Preview 1:开启下一代开发框架新篇章
· C# 集成 DeepSeek 模型实现 AI 私有化(本地部署与 API 调用教程)
· DeepSeek R1 简明指南:架构、训练、本地部署及硬件要求
· 2 本地部署DeepSeek模型构建本地知识库+联网搜索详细步骤
点击右上角即可分享
微信分享提示