23种设计模式之——单例模式
什么是单例模式?
保证一个类仅有一个实例,并提供一个访问它的全局访问点。
什么时候用到它呢?
通常我们可以让一个全局变量使得一个对象被访问,但它不能防止你实例化多个对象。
一个最好的方法就是,让类自身负责保存它的唯一实例。
这个类可以保证没有其他实例可以被创建,并且它可以提供一个访问该实例的方法。
单例模式有什么好处呢?
1、单例模式可以保证唯一的实例。
2、单例模式因为singleton类封装它的唯一实例,这样它可以严格地控制客户怎样访问它以及何时访问它。简单地说就是对唯一实例的受控访问。
单例模式分为饿汉单例模式和懒汉单例模式。
饿汉单例模式:静态初始化的方式是在自己被加载时就将自己实例化,所以被形象地称之为饿汉式单例模式。
懒汉单例模式:要在第一次被引用时,才会将自己实例化,所以就被称为懒汉式单例模式。
懒汉单例模式实现:
singleton类,定义一个getinstance操作,允许客户访问它的唯一实例。getinstance是一个静态方法,主要负责创建自己的唯一实例。
构造方法让其private,这就堵死了外界利用new创建其实例的可能。
getinstance()方法是获得本类实例的唯一全局访问点。
若实例不存在,则new一个新实例,否则返回已有的实例。
客户端代码:
运行结果为:
比较两次实例化后对象的结果是实例相同。
多线程时的单例
在多线程的程序中,多个线程同时,注意是同时访问singleton类,调用getinstance()方法,会有可能造成创建多个实例的。
如何解决呢?
可以给进程一把锁来处理。synchronized是确保当一个线程位于代码的临界区时,另一个线程不进入临界区。如果其他线程试图进入锁定的代码,则它将一直等待(即被阻止),直到该对象被释放。
代码实现:
由于有了synchronized,就保证了多线程环境下的同时访问也不会造成多个实例的生成。
每次调用getinstance()都需要synchronized,这种做法会印象性能,所以还需要对该类进行改良。
改良之后的代码:
先判断实例是否存在,不存在再加锁处理。
我们在外面已经判断能力instance实例是否存在,为什么在synchronized里面还需要在做一次instance实例是否存在的判断呢?
如果有两个线程调用getinstance()方法时,它们将都可以通过第一重instance==null的判断。
由于synchronized机制,这两个线程只有一个进入,另一个在排队等待,必须要其中一个进入并出来后,另一个才能进入。
但是此时如果没有了第二重instance是否为null的判断,则第一个线程创建了实例,而第二个线程还是可以继续再创建新的实例,这样就没有达到单例的目的。
饿汉单例模式实现:
饿汉单例模式的加载方式是在自己被加载时就将自己实例化了。
喜欢的小伙伴们可以搜索我们个人的微信公众号“程序员的成长之路”点击关注或扫描下方二维码
上一篇: 详解C#委托和事件(一)