Java单例模式

  1. 简单介绍

      在刚开始接触Java的构造方法的时候,看到构造方法用的都是public,突然想如果用private修饰会怎样,试了一下,创建对象的时候报错,例子如下图:
    

    这里写图片描述

       上网查了查,发现的确有这种private的构造方法,叫做单例模式。构造函数弄成private 就是单例模式,即不想让别人用new 方法来创建多个对象,可以在类里面先生成一个对象,然后写一个public static方法把这个对象return出去。
    

    单例模式有以下特点:
      1、单例类只能有一个实例。
      2、单例类必须自己创建自己的唯一实例。
      3、单例类必须给所有其他对象提供这一实例。
      

  2. 几种常见的单例模式

    1. 饿汉式

      //静态方法只在编译期间执行一次初始化,以后不再改变,也就是只会有一个对象。线程安全。
      public class Singleton1 {  
          private Singleton1() {}  
          private static final Singleton1 single = new Singleton1();  
      
          public static Singleton1 getInstance() {  
              return single;  
          }  
      }  
    2. 懒汉式

      一次判断

      //那个if判断确保对象只创建一次,并加上synchronized保证线程安全
      public class Singleton2 {  
          private Singleton2() {}  
          private static Singleton2 single = null;  
      
          public static synchronized Singleton2 getInstance() {  
               if (single == null) {    
                   single = new Singleton();  
               }    
              return single;  
      
      }  

      两次判断

      //在getInstance中做了两次null检查,确保了只有第一次调用单例的时候才会做同步,这样也是线程安全的,同时避免了每次都同步的性能损耗
      public static Singleton2 getInstance() {  
              if (single == null) {    
                  synchronized (Singleton2.class) {    
                     if (single == null) {    
                        singleton = new Singleton2();   
                     }    
                  }    
              }    
              return singleton;   
          }  
    3. 登记式单例类

      //登记式单例实际上维护了一组单例类的实例,将这些实例存放在一个Map(登记薄)中,对于已经登记过的实例,则从Map直接返回,对于没有登记的,则先登记,然后返回。 
      //类似Spring里面的方法,将类名注册,下次从里面直接获取。  
      public class Singleton3 {  
          private static Map<String,Singleton3> map = new HashMap<String,Singleton3>();  
          static{  
              Singleton3 single = new Singleton3();  
              map.put(single.getClass().getName(), single);  
          }  
          //保护的默认构造子  
          protected Singleton3(){}  
          //静态工厂方法,返还此类惟一的实例  
          public static Singleton3 getInstance(String name) {  
              if(name == null) {  
                  name = Singleton3.class.getName();  
                  System.out.println("name == null"+"--->name="+name);  
              }  
              if(map.get(name) == null) {  
                  try {  
                      map.put(name, (Singleton3) Class.forName(name).newInstance());  
                  } catch (InstantiationException e) {  
                      e.printStackTrace();  
                  } catch (IllegalAccessException e) {  
                      e.printStackTrace();  
                  } catch (ClassNotFoundException e) {  
                      e.printStackTrace();  
                  }  
              }  
              return map.get(name);  
          }  
          //一个示意性的商业方法  
          public String about() {      
              return "Hello, I am RegSingleton.";      
          }      
          public static void main(String[] args) {  
              Singleton3 single3 = Singleton3.getInstance(null);  
              System.out.println(single3.about());  
          }  
      } 
  3. 小总结

    • 饿汉式在类创建的同时就实例化一个静态对象出来,不管之后会不会使用这个单例,都会占据一定的内存,但是相应的,在第一次调用时速度也会更快,因为其资源已经初始化完成,

      而懒汉式顾名思义,会延迟加载,在第一次使用该单例的时候才会实例化对象出来,第一次调用时要做初始化,如果要做的工作比较多,性能上会有些延迟,之后就和饿汉式一样了。

参考文献:http://blog.youkuaiyun.com/jason0539/article/details/23297037/

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值