Java单例的7种写法

1. 懒汉式(线程不安全)

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

2. 懒汉式(线程安全)

class Singleton{
  private static Singleton instance=null;
  public static synchronized Singleton getInstance(){
    if(instance==null){
      instance=new Instance();
    }
    return instance;
  }
}
  • 优点:线程安全;
  • 缺点:加了同步,效率低;

3. 饿汉式

class Singleton{
  private static final Singleton INSTANCE = new Singleton();
  public static Singleton getInstance(){
    return INSTANCE;
  }
}
  • 特点:在类初始化的时候就加载了instance;

4. 饿汉式变种

class Singleton{
  private Singleton instance=null;  
  static{
      instance=new Instance();
  } 
  public static Singleton getInstance(){
    return instance;
  }
}
  • 特点:在类初始化的时候就加载了instance;
    本质上第四种方法和第三种方法一样;

5. 静态内部类

class Singleton{
  private static class SingletonHolder{
    private static Singleton instance=new Singleton();
  }
  public static Singleton getInstance(){
    return SingletonHolder.instance;
  }
}
  • 特点:类级的内部类,只有被调用的时候才会被加载,从而实现了延迟加载;
  • 优点:懒加载;

6. 枚举

public enum Singleton{
    INSTANCE;
    public void whateverMethod(){
  }
}
  • 特点:不仅能避免多线程同步问题,还能防止反序列化重新创建对象;
  • effect java 推荐方式。enum在jdk1.5 才引入。

7. 双重校验法

class Singleton{
  private volatile static Singleton instance=null;
  public static Singleton getInstance(){
    if(instance!=null){
      return instance;
    }
    synchronized(Singleton.class){
      if(instance==null){
        instance=new Instance();
      }
      return instance;
    }
  }
}
  • 特点:只有在jdk1.5之后,双重检验锁才能正常达到单例效果;

总结
有两个问题需要注意:
1.如果单例由不同的类装载器装入,那便有可能存在多个单例类的实例。假定不是远端存取,例如一些servlet容器对每个servlet使用完全不同的类装载器,这样的话如果有两个servlet访问一个单例类,它们就都会有各自的实例。
2.如果Singleton实现了java.io.Serializable接口,那么这个类的实例就可能被序列化和复原。不管怎样,如果你序列化一个单例类的对象,接下来复原多个那个对象,那你就会有多个单例类的实例。
对第一个问题修复的办法是:

private static Class getClass(String classname)      
                                        throws ClassNotFoundException {     
     ClassLoader classLoader = Thread.currentThread().getContextClassLoader();     
     
     if(classLoader == null)     
        classLoader = Singleton.class.getClassLoader();     
     
     return (classLoader.loadClass(classname));     
  }     
}  

对第二个问题修复的办法是:

public class Singleton implements java.io.Serializable {     
  public static Singleton INSTANCE = new Singleton();     
     
  protected Singleton() {     
       
  }     
  private Object readResolve() {     
           return INSTANCE;     
     }    
}   
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,419评论 11 349
  • 单例模式(SingletonPattern)一般被认为是最简单、最易理解的设计模式,也因为它的简洁易懂,是项目中最...
    成热了阅读 4,324评论 4 34
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,867评论 18 399
  • 掌握技巧,可以和任何人都聊得来。 参与到谈话中 如果交谈中只有对方在说,交谈的内容就会很有限。 每个人的性格不同,...
    wukaili阅读 191评论 0 1
  • 但妈妈早上好,我们今天一起来聊一下,关于学习和获取靠谱知识的渠道。其实这个话题对我来说也是非常困难的。因为我们其实...
    丁铃铛阅读 242评论 0 0