Java单例模式

来源:互联网 发布:那个人好像一条狗 知乎 编辑:程序博客网 时间:2024/06/03 20:37
  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.csdn.net/jason0539/article/details/23297037/

0 0