枚举实现单例模式

1.引言

单元素的枚举类型已经成为实现Singleton的最佳方法
                      -- 出自 《effective java》

2.单例模式的特点

  • 单例模式三个主要特点:
    1、构造方法私有化;
    2、实例化的变量引用私有化;
    3、获取实例的方法共有。

3. 常用的单例模式

1.单例的饿汉模式

 1   public class Singleton {
 2     /*
 3      * 利用静态变量来记录Singleton的唯一实例
 4      * 直接初始化静态变量,这样就可以确保线程安全了
 5      */
 6     private static Singleton uniqueInstance = new Singleton();
 7
 8     /*
 9      * 构造器私有化,只有Singleton类内才可以调用构造器
10      */
11     private Singleton(){
12
13     }
14
15     public static Singleton getInstance(){
16         return uniqueInstance;
17     }
18
19 }

2.懒汉的双重加锁机制

 1 public class Singleton {
 2     /*
 3      * 利用静态变量来记录Singleton的唯一实例
 4      * volatile 关键字确保:当uniqueInstance变量被初始化成Singleton实例时,
 5      * 多个线程正确地处理uniqueInstance变量
 6      *
 7      */
 8     private volatile static Singleton uniqueInstance;
 9
10     /*
11      * 构造器私有化,只有Singleton类内才可以调用构造器
12      */
13     private Singleton(){
14
15     }
16
17     /*
18      *
19      * 检查实例,如果不存在,就进入同步区域
20      */
21     public static Singleton getInstance(){
22         if(uniqueInstance == null){
23             synchronized(Singleton.class){    //进入同步区域
24                 if(uniqueInstance == null){     //在检查一次,如果为null,则创建
25                     uniqueInstance  = new Singleton();
26                 }
27             }
28         }
29
30         return uniqueInstance;
31     }
32
33 }

3.静态内部类

public class Singleton {    
    private static class LazyHolder {    
       private static final Singleton INSTANCE = new Singleton();    
    }    
    private Singleton (){}    
    public static final Singleton getInstance() {    
       return LazyHolder.INSTANCE;    
    }    
}

4.为什么使用单例

4.1 私有化构造器并不保险

    《effective java》中只简单的提了几句话:“享有特权的客户端可以借助AccessibleObject.setAccessible方法,通过反射机制调用私有构造器。如果需要低于这种攻击,可以修改构造器,让它在被要求创建第二个实例的时候抛出异常。

4.2序列化问题

    任何一个readObject方法,不管是显式的还是默认的,它都会返回一个新建的实例,这个新建的实例不同于该类初始化时创建的实例。”当然,这个问题也是可以解决的,想详细了解的同学可以翻看《effective java》第77条:对于实例控制,枚举类型优于readResolve

4.3 枚举单例示例

public enum  EnumSingleton {
    INSTANCE;
    public EnumSingleton getInstance(){
        return INSTANCE;
    }
}

单例模式


public class User {
    //私有化构造函数
    private User(){ }
 
    //定义一个静态枚举类
    static enum SingletonEnum{
        //创建一个枚举对象,该对象天生为单例
        INSTANCE;
        private User user;
        //私有化枚举的构造函数
        private SingletonEnum(){
            user=new User();
        }
        public User getInstnce(){
            return user;
        }
    }
 
    //对外暴露一个获取User对象的静态方法
    public static User getInstance(){
        return SingletonEnum.INSTANCE.getInstnce();
    }
}

public class Test {
    public static void main(String [] args){
        System.out.println(User.getInstance());
        System.out.println(User.getInstance());
        System.out.println(User.getInstance()==User.getInstance());
    }
}
结果为true

以上代码看起来已经是ok了,其实不是,可能还存在反射攻击或者反序列化攻击

最终版

public enum Singleton {

    INSTANCE;

    public void doSomething() {
        System.out.println("doSomething");
    }

}
调用方法:

public class Main {

    public static void main(String[] args) {
        Singleton.INSTANCE.doSomething();
    }

}

直接通过Singleton.INSTANCE.doSomething()的方式调用即可。方便、简洁又安全。

5.总结

至此,相信大家应该能明白了为什么Joshua Bloch说的“单元素的枚举类型已经成为实现Singleton的最佳方法”了吧。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 枚举单例(Enum Singleton)是实现单例模式的一种新方式,尽管单例模式在java中已经存在很长时间了,但...
    淡淡的伤你阅读 1,183评论 0 3
  • 枚举实现单例模式 前面我们说到序列化和反序列化以及反射对单例都是有破坏的,下面我们介绍一种更加优雅的实现,也是ef...
    三不猴子阅读 3,478评论 0 1
  • 单例模式(SingletonPattern)一般被认为是最简单、最易理解的设计模式,也因为它的简洁易懂,是项目中最...
    成热了阅读 4,292评论 4 34
  • 微信原文:设计模式 | 单例模式及典型应用 单例是最常见的设计模式之一,实现的方式非常多,同时需要注意的问题也非常...
    小旋锋的简书阅读 1,791评论 2 5
  • 前言 EFK可能都不熟悉,实际上EFK是大名鼎鼎的日志系统ELK的一个变种 在没有分布式日志的时候,每次出问题了需...
    Raye阅读 30,201评论 0 28