C# 单例模式(Singleton Pattern)

(新手写博客,主要是对自己学习的归纳总结。会对很多小细节详解。)

单例模式的定义:

确保一个类只有一个实例,并提供一个全局访问点。

首先实例大家应该都明白就是类生成对象的过程简单的就是String s=new String(),则s就是个实例。

Q:如何只生成一个实例?

A:1)首先必须将构造函数变为私有从而防止其他类实例化,并且只能有一个构造函数。因为系统会默认一个无参构造函数,而且默认public访问修饰符。 所以必须写一个私有无参让默认无效。(通常单例模式都是不带形参的)

2)在该类中声明一个自己本身的静态实例,然后通过静态方法返回。

Q:如何提供一个全局访问点?

A:在类中创建一个公共并且静态的属性。(因为静态方法是类中的一个成员方法,属于整个类,即不用创建任何对象也可以直接调用。单例模式是不允许其他类实例的。)

代码:

分为两种模式:

1.LAZY模式

就是延迟加载, 设计模式是为了避免一些无谓的性能开销而提出来的,所谓延迟加载就是当在真正需要数据(读取属性值)的时候,才真正执行数据加载操作.有效使用它可以大大提高系统性能。

2.饿汉模式

与LAZY模式相反 ,加载时会将自己实例化。起来最容易的单例模式。

分析代码1:(经典)

 1 // 不要用这种方式
 2 public sealed class Singleton
 3 {
 4     private static Singleton instance=null;//声明自己本身的静态实例
 5     private Singleton(){}//私有构造
 6     public static Singleton Instance //提供全局访问点
 7     {
 8             if (instance==null)//实例不存在则创建
 9             {
10                 instance = new Singleton();
11             }
12             return instance;
13     }
14 }

该代码仅供理解,单例模式的定义。

问题:该方法是非线程安全的,当有两个线程同时进入时,如果instance为null则都会创建实例。实际上,在测试以前,实例就已经有可能被创建了,但是内存模型不能保证这个实例能被其他的线程看到。

下面我们优化改进

分析代码2:(非安全线程)

public sealed class Singleton
{
    private static Singleton instance = null;
    private static readonly object padlock = new object();//定义一个标识确保线程同步
    Singleton(){}
    public static Singleton Instance
    {
            lock (padlock)//线程到达时加锁 运行完之后解锁 当遇到加锁线程就会挂起等待解锁
            {
                if (instance == null)
                {
                    instance = new Singleton();
                }
                return instance;
            }
    }
}

以上解决了多线程问题。

问题:性能上来说,锁变成了每次都必须的当这个实例被响应的时候。此时完全没必要对线程辅助对象加锁之后再去判断,所以上面的实现方式增加了额外的开销。

下面我们进行优化改进:

 代码分析3:(双重锁定)

 1 public sealed class Singleton
 2 {
 3     private static Singleton instance = null;
 4     private static readonly object padlock = new object();
 5     Singleton(){}
 6     public static Singleton Instance
 7     {
 8         get
 9         {
10             if (instance == null)//外层的if语句块,这使得每个线程欲获取实例时不必每次都得加锁,因为只有实例为空时(即需要创建一个实例),才需加锁创建
11             {
12                 lock (padlock)
13                 {
14                     if (instance == null)
15                     {
16                         instance = new Singleton();
17                     }
18                 }
19             }
20             return instance;
21         }
22     }
23 }

这种“双重检查锁定”理论上是完美的

问题是:并不能保证它会在单处理器或多处理器计算机上顺利运行。(反正就是有问题吧 之后再研读一下 看看具体是怎么回事)

代码分析4:(不完全LAZY)

 1 public sealed class Singleton
 2 {
 3     private static readonly Singleton instance = new Singleton();
 4
 5     // 显示的static 构造函数
 6     //静态构造函数抑制了beforefieldinit 特性(访问成员之前就执行静态函数)
 7     static Singleton(){}
 8     private Singleton(){}
 9     public static Singleton Instance
10     {
11         get
12         {
13             return instance;
14         }
15     }
16 }

不完全LAZY模式(通过抑制beforefildinit特性并不能起到太大的效果)

代码分析5:(完全LAZY)

 1 public sealed class Singleton
 2 {
 3     private Singleton(){}
 4     public static Singleton Instance { get { return Nested.instance; }}
 5 //嵌套类
 6     private class Nested
 7     {
 8        //抑制beforefieldinit特性
 9         static Nested(){}
10         internal static readonly Singleton instance = new Singleton();
11     }
12 }

这里使用了嵌套类(嵌套类型是LAZY加载的,也就是说嵌套类型在使用他时才会初始化)

代码分析6:(Lazy<T>)

1 public sealed class Singleton
2 {
3 //使用.NET4 Lazy<T>
4     private static readonly Lazy<Singleton> lazy =new Lazy<Singleton>(() => new Singleton());
5     public static Singleton Instance { get { return lazy.Value; } }
6     private Singleton() {}
7 }

Lazy<T> 对象初始化默认是线程安全的,在多线程环境下,第一个访问 Lazy<T> 对象的 Value 属性的线程将初始化 Lazy<T> 对象,以后访问的线程都将使用第一次初始化的数据。

以上全部是LAZY模式,现在了解下饿汉模式

代码分析7:

public sealed class Singleton
{
        private static readonly Singleton instance=new Singleton();//直接实例化
        private Singleton(){}
        public static Singleton Instance()
        {
               return instance;
        }
}

在这种模式下,无需自己解决线程安全性问题,CLR会给我们解决。由此可以看到这个类被加载时,会自动实例化这个类,而不用在第一次调用Instance()后才实例化出唯一的单例对象。

为了优化系统当然还是选择优化模式。LAZY模式最好的应该是使用Lazy<T>简短安全。

以上都是我在博客园对单例模式学习的总结。今后会补充一些具体的项目中的案例让和我一样的新手更容易吸收理解最终达到举一反三。这里重点参考了这里的文章,希望大家有指教的地方多多提点。感谢。

时间: 2024-12-06 21:59:34

C# 单例模式(Singleton Pattern)的相关文章

二十四种设计模式:单例模式(Singleton Pattern)

单例模式(Singleton Pattern) 介绍保证一个类仅有一个实例,并提供一个访问它的全局访问点. 示例保证一个类仅有一个实例. Singleton using System; using System.Collections.Generic; using System.Text; namespace Pattern.Singleton { /// <summary> /// 泛型实现单例模式 /// </summary> /// <typeparam name=&q

ninject学习笔记二:单例模式singleton pattern

今天准备学习singleton pattern,顾单词思含义,就是一个实例的意思.单例的实现思路是:私有化构造函数,提供公有方法获取类的实例.下面定义一个音乐播放器类MusicPlayer,观察单例的实现方法 class MusicPlayer { private static readonly MusicPlayer player = new MusicPlayer(); private MusicPlayer() { } public static MusicPlayer GetInstan

设计模式之单例模式Singleton pattern

单例模式Singleton pattern 一种软件设计模式.在核心结构中只包含一个被称为单例的特殊类. 一个类只有一个对象实例,并且自行实例化向整个系统提供. 动机 一个系统中可以存在多个打印任务,但是只能有一个正在工作的任务:一个系统只能有一个窗口管理器或文件系统:一个系统只能有一个计时工具或ID(序号)生成器.如在Windows中就只能打开一个任务管理器.如果不使用机制对窗口对象进行唯一化,将弹出多个窗口,如果这些窗口显示的内容完全一致,则是重复对象,浪费内存资源:如果这些窗口显示的内容不

单例模式 (Singleton pattern)

What is Singleton pattern? In Wikipedia, there is an explanation:"In software engineering, the singleton pattern is a design pattern that restricts the instantiation of a class to one object." 一.什么是单例模式? 在维基百科中,是这样解释的,“在软件工程中,单例模式指的是对类加以限制,只允许创建

.NET设计模式实例之单例模式( Singleton Pattern)

一.单例模式简介(Brief Introduction) 单例模式(Singleton Pattern),保证一个类只有一个实例,并提供一个访问它的全局访问点.单例模式因为Singleton封装它的唯一实例,它就可以严格地控制客户怎样访问它以及何时访问它. 二.解决的问题(What To Solve) 当一个类只允许创建一个实例时,可以考虑使用单例模式. 三.单例模式分析(Analysis)1.单例模式结构 Singleton类,定义一个私有变量instance;私有构造方法Singleton(

Swift 实现单例模式Singleton pattern的三种方法

转自:点击打开链接 From my short experience with Swift there are three approaches to implement the Singleton pattern that support lazy initialization and thread safety. These approaches might change or become redundant as the language matures. Global constant

设计模式之五:单例模式(Singleton Pattern)

单例模式的意思就是只有一个实例.单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例.这个类称为单例类. 单例模式有三个要点: 点某个类只能有一个实例: 它必须自行创建这个实例 它必须自行向整个系统提供这个实例. 一些资源管理器常常设计成单例模式. C#中的单例模式 保证一个类仅有一个实例,并提供一个访问它的全局访问点 实现要点: 单例模式是限制而不是改进类的创建. 单例模式一般不要支持Icloneable接口,因为这可能导致多个对象实例,与单例模式的初衷违背. 单例模式一般

&quot;围观&quot;设计模式(7)--创建型之单例模式(Singleton Pattern)

单例模式,也叫单子模式,是一种常用的软件设计模式.在应用这个模式时,单例对象的类必须保证只有一个实例存在.许多时候整个系统只需要拥有一个的全局对象,这样有利于我们协调系统整体的行为.比如在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再通过这个单例对象获取这些配置信息.这种方式简化了在复杂环境下的配置管理.----维基百科(WIKIPEDIA) 个人的理解: 单例模式概念比较简单,他的目的就是只允许出现一个该类的实例,经常在JD

[设计模式]&lt;3&gt;. C++与单例模式(singleton pattern)

默默地EEer 原文地址: http://www.cnblogs.com/hebaichuanyeah/p/5608209.html 单例模式是指,保证一个类,只有唯一一个实例,并提供这个实例的全局访问. 1.不靠谱之单例模式 很容易写出如下代码,通过Singleton类中的static函数去获取唯一的static指针变量,同时把Singalton类的构造函数设置为private,不允许构造多个对象. #include "iostream" using namespace std; c