Java设计模式之单例模式

单例模式定义:

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

单例模式有什么用处?

    有一些对象只需要一个,如Windows是多进程多线程的,在操作一个文件的时候,就不可避免的出现多个进程或线程同时操作一个文件的现象,所以所有文件的处理必须通过唯一的实例来进行。一些设备管理器常常设计为单例模式,如一个电脑有两台打印机,在输出的时候就要处理不能两台打印机打印同一个文件。

单例模式提供全局访问点和全局变量的比较:

如果将对象赋值给一个全局变量,就必须在程序的一开始就创建好对象,如果这个对象非常耗费资源,而程序在执行过程中又一直没有用到它,就会造成资源的浪费。而利用单例模式,可以在需要时才创建对象

单例模式类图:


单例模式有多种写法,它们各有利弊:

1.饿汉模式

public class Singleton {  
     private static Singleton instance = new Singleton();  
     private Singleton (){
     }
     public static Singleton getInstance() {  
     return instance;  
     }  
 }  

这种方式因为单例的实例被声明为static 和final变量了,所以在类加载时就完成了初始化,这种方式基于类加载机制避免了多线程的同步问题。但是没有达到在需要对象的时候才去创建它的效果

2. 懒汉模式(线程不安全)

public class Singleton {  
      private static Singleton instance; //利用一个静态变量来记录Singleton类的唯一实例
      private Singleton (){                //把构造器声明为私有的,只有在Singleton类才能调用构造器
      }   
      public static Singleton getInstance() {  
      if (instance == null) {              //判断instance是否为空,为空则还没有创建实例
          instance = new Singleton();  
      }  
      return instance;          //如果不为空,表示之前已经创建过对象,直接返回之前的对象
      }  
 } 

这种方式在用户第一次调用getInstance()方法时创建对象,虽然节约了资源,但是在多线程不能正常工作。(如果有多个线程都要执行这段代码,线程1先判断instance是否为空,在线程1还没有执行instance = new Singleton()创建实例对象时,线程2也判断instance是否为空,此时就会造成实例化多个对象)。

3.懒汉模式(线程安全)

public class Singleton {  
      private static Singleton instance;  
      private Singleton (){
      }
      public static synchronized Singleton getInstance() {  
      if (instance == null) {  
          instance = new Singleton();  
      }  
      return instance;  
      }  
 }

这种方式将getInstance()变成同步(synchronized)方法,可以在多线程中很好的工作,但是每次调用getInstance()方法时都需要进行同步,造成不必要的同步开销。

4. 双重检查模式 (DCL)

利用双重检查加锁(double-checked locking),首先检查是否实例已经创建了,如果尚未创建,才进行同步,这样只有第一次会同步,提高了效率。

public class Singleton {  
      private volatile static Singleton singleton;  
      private Singleton (){
      }   
      public static Singleton getInstance() {  
      if (instance== null) {  
          synchronized (Singleton.class) {  
          if (instance== null) {  
              instance= new Singleton();  
          }  
         }  
     }  
     return singleton;  
     }  
 }  

这里在getInstance()方法中对instance进行了两次判空,第一次检查实例,如果不存在进入同步区域,第二次,进入同步区域后再检查一次,如果仍然是null,才创建实例,资源利用率高。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值