• 有些类也需计划生育 单例模式


    21.1 类也需计划生育

    21.2 判断对象是否是null

    namespace 单例模式
    {
        public partial class Form1 : Form
        {
            public Form1()
            {
                InitializeComponent();
            }
    
            private void Form1_Load(object sender, EventArgs e)
            {
                this.IsMdiContainer = true;
            }
    
            private FormToolbox ftb;                             //类变量声明,
    
            private void ToolStripMenuItemToolbox_Click(object sender, EventArgs e)
            {
                if (ftb == null || ftb.IsDisposed)              //关闭工具箱时,实例并没有变为null而只是Disposed,
                {
                    ftb = new FormToolbox();
                    ftb.MdiParent = this;
                    ftb.Show();
                }
            }
    
        }
    }
    View Code
    namespace 单例模式
    {
        public partial class Form1 : Form
        {
            public Form1()
            {
                InitializeComponent();
            }
    
            private void Form1_Load(object sender, EventArgs e)
            {
                this.IsMdiContainer = true;
            }
    
            private FormToolbox ftb;
    
            private void ToolStripMenuItemToolbox_Click(object sender, EventArgs e)
            {
                openToolbox();
            }
    
            private void toolStripButton1_Click(object sender, EventArgs e)
            {
                openToolbox();
            }
    
            private void openToolbox()
            {
                if (ftb == null || ftb.IsDisposed)
                {
                    ftb = new FormToolbox();
                    ftb.MdiParent = this;
                    ftb.Show();
                }
            }
    
        }
    }
    View Code

    21.3 生还是不生是自己的责任

    所有类都有构造函数,不编码则系统默认生成空的构造函数,若有显示定义的构造函数,默认的构造函数就会失效,将类的构造函数写成private的,外部程序就不能用new来实例化它了,

    namespace 单例模式
    {
        public partial class FormToolbox : Form
        {
            private static FormToolbox ftb = null;           //声明一个静态类变量,         
    
            private FormToolbox()                            //构造函数私有,外部代码不能直接new来实例化,         
            {
                InitializeComponent();
            }
    
            public static FormToolbox GetInstance()          //得到类实例的方法,返回值就是本类对象,注意也是静态的         
            {
                if (ftb == null || ftb.IsDisposed)
                {
                    ftb = new FormToolbox();                 //当内部的ftb是null或被Dispose过,则new它,
                    ftb.MdiParent = Form1.ActiveForm;        //并且设计其Mdiparent为Form1,此时将实例化的对象存在静态变量ftb中,
                }                                            //以后就可以不用实例化得到它了,
                return ftb;
            }
    
        }
    }
    View Code
    namespace 单例模式
    {
        public partial class Form1 : Form
        {
            public Form1()
            {
                InitializeComponent();
            }
    
            private void Form1_Load(object sender, EventArgs e)
            {
                this.IsMdiContainer = true;
            }
    
            private void ToolStripMenuItemToolbox_Click(object sender, EventArgs e)
            {
                FormToolbox.GetInstance().Show();
            }
    
            private void toolStripButton1_Click(object sender, EventArgs e)
            {
                FormToolbox.GetInstance().Show();
            }
    
        }
    }
    View Code

    这样一来,客户端不在考虑是否需要去实例化的问题,而把责任都给了应该负责的类去处理,其实这就是单例模式,

    21.4 单例模式

    保证一个类仅有一个实例,并提供一个访问它的全局访问点,

    通常我们可以让一个全局变量使得一个对象被访问,但它不能防止你实例化多个对象,一个最好的办法就是,让类自身负责保存它的唯一实例,这个类可以保证没有其它实例可以被创建,并且它可以提供一个访问该实例的方法,

    namespace 单例模式
    {
        class Program
        {
            static void Main(string[] args)
            {
                Singleton s1 = Singleton.GetInstance();
                Singleton s2 = Singleton.GetInstance();
    
                if (s1 == s2)                                      //比较两次实例化后对象的结果是实例相同,
                {
                    Console.WriteLine("Objects are the same instance");
                }
    
                Console.Read();
            }
        }
    
    
        class Singleton
        {
            private static Singleton instance;
    
            private static readonly object syncRoot = new object();
    
            private Singleton()                                    //构造函数让其private,堵死外界利用new创建实例的可能,
            {
    
            }
    
            public static Singleton GetInstance()                  //此方法是获得本类实例的唯一全局访问点,
            {
                if (instance == null)                              //若实例不存在,则new一个新实例,否则返回已有实例,
                {
                    instance = new Singleton();
                }
                return instance;
            }
        }
    View Code

    单例模式除了可以保证唯一的实例外,它还可以严格的控制客户怎样访问它以及何时访问它,简单的说就是对唯一实例的受控访问,

    21.5 多线程时的单例

    多线程的程序中,多个线程同时访问Singleton类,调用GetInstance方法,会有可能造成创建多个实例,

    可以给进程一把锁来处理,lock是确保当一个线程位于代码的临界区时,另一个线程不能进入临界区,如果其它线程试图进入锁定的代码,则它将一直等待即被阻止,直到该对象被释放,

    class Singleton
    {
        private static Singleton instance;
    
        private static readonly object syncRoot = new object();//加锁时,instance实例有没有被创建过都还不知道,无法对其加锁,
    
        private Singleton()                                    //构造函数让其private,堵死外界利用new创建实例的可能,
        {
    
        }
    
        public static Singleton GetInstance()                  //此方法是获得本类实例的唯一全局访问点,
        {
            lock (syncRoot)                                    //在同一时刻加了锁的那部分程序只有一个线程可以进入,
            {
                if (instance == null)                          //若实例不存在,则new一个新实例,否则返回已有实例,
                {
                    instance = new Singleton();
                }
            }
            return instance;
        }
    }
    View Code

    21.6 双重锁定

    class Singleton
    {
        private static Singleton instance;
    
        private static readonly object syncRoot = new object();//加锁时,instance实例有没有被创建过都还不知道,无法对其加锁,
    
        private Singleton()                                    //构造函数让其private,堵死外界利用new创建实例的可能,
        {
    
        }
    
        public static Singleton GetInstance()                  //此方法是获得本类实例的唯一全局访问点,
        {
            if (instance == null)                              //先判断实例是否存在,不存在再加锁处理,
            {
                lock (syncRoot)                                //在同一时刻加了锁的那部分程序只有一个线程可以进入,
                {
                    if (instance == null)                      //若实例不存在,则new一个新实例,否则返回已有实例,
                    {
                        instance = new Singleton();
                    }
                }
            }
            return instance;
        }
    }
    View Code

    当instance为nul并且同时有两个线程调用GetInstance方法时,它们都可以通过第一重instance==null的判断,然后由于lock机制,两个线程则只有一个进入,另一个在外排队等候,必须要其中的一个进入并出来后,另一个才能进入,而此时如果没有了第二重的instance是否判断为null的判断,则第一个线程创建了实例,而第二个线程还是可以继续再创建新的实例,这就没有达到单例的目的,

    21.7 静态初始化

    C#与公共语言运行库也提供了一种“静态初始化”方法,这种方法不需要开发人员显示的编写线程安全代码,即可解决多线程环境下它是不安全的问题,

    public sealed class Singleton                                      //阻止发生派生,而派生可能会增加实例,
    {
        private static readonly Singleton instance = new Singleton();  //在第一次引用类的任何成员时创建实例,公共语言运行库负责处理变量初始化,
    
        private Singleton() { }
    
        public static Singleton GetInstance()
        {
            return instance;
        }
    }
    View Code

    这样的实现与前面的示例类似,也是解决了单例模式试图解决的两个基本问题,全局访问和实例化控制,公共静态属性为访问实例提供了一个全局的访问点,不同之处在于它依赖公共语言运行库来初始化变量,由于构造函数时私有的,因此不能在类本身以外实例化Singleton类,因此,变量引用的是可以在系统中唯一存在的实例,注意instance变量标记为readonly。意味着只能在静态初始化期间或在类构造函数中分配变量,由于静态初始化的方式是在自己被加载时就将自己实例化,所以被形象称为饿汉单例模式,原先的单例模式处理方式是要在第一次被引用时,才会将自己实例化,所以称之为懒汉单例模式,

    其区别为,饿汉式即静态初始化的方式,它是类一加载就实例化的对象,所以要提前占用系统资源,而懒汉式又会面临着多线程访问的安全性问题,需要双重锁定这样的处理才可以保证安全,

  • 相关阅读:
    尚硅谷面试第一季-05递归与迭代
    尚硅谷面试第一季-04方法的参数传递机制
    尚硅谷面试第一季-03类初始化和实例初始化
    python爬爬爬之单网页html页面爬取
    python之花瓣美女下载
    (转载博文)VC++API速查
    (转载博文)MFC 窗口句柄获取
    全局变量的声明
    python图片小爬虫
    Opencv2.4.4作图像旋转和缩放
  • 原文地址:https://www.cnblogs.com/huangxuQaQ/p/11304808.html
Copyright © 2020-2023  润新知