设计模式之单例模式(Java实现)(二)

简介: 设计模式之单例模式(Java实现)(二)

三、枚举类实现单例(解决反射安全问题)


查看源码


为什么我们通过使用枚举类能够实现单例呢?


通过看反射方法newInstance()(Constructor类)的源码。
@CallerSensitive
public T newInstance(Object ... initargs)
    throws InstantiationException, IllegalAccessException,
           IllegalArgumentException, InvocationTargetException
{
    if (!override) {
        if (!Reflection.quickCheckMemberAccess(clazz, modifiers)) {
            Class<?> caller = Reflection.getCallerClass();
            checkAccess(caller, clazz, null, modifiers);
        }
    }
    //如果是枚举类型直接,直接抛出异常,不让创建实例对象
    if ((clazz.getModifiers() & Modifier.ENUM) != 0)
        throw new IllegalArgumentException("Cannot reflectively create enum objects");
    ConstructorAccessor ca = constructorAccessor;   // read volatile
    if (ca == null) {
        ca = acquireConstructorAccessor();
    }
    @SuppressWarnings("unchecked")
    T inst = (T) ca.newInstance(initargs);
    return inst;
}


通过源码我们可以知道对于枚举类型无法使用newInstance来创建实例,也就能够解决单例在反射中的安全问题!



测试过程


见single目录下的OwnStudent:这是后来补上去的仅仅是类名不相同而已



自定义枚举类:


public enum Student {
    STUDENT;
}


我们来实验一下即可知道了,首先来若是想要通过反射来newInstance首先需要获取Constructor构造器实例,所以我们最起码得先知道其构造器的参数以及类型吧,我们通过反编译Student.class文件来查看一下:


①IDEA中的反编译


public enum Student {
    STUDENT;
    private Student() {
    }
}


②JDK工具javap进行反编译



③使用jad工具来进行反编译,获取一个java文件如下:


public final class Student extends Enum
{
    public static Student[] values()
    {
        return (Student[])$VALUES.clone();
    }
    public static Student valueOf(String name)
    {
        return (Student)Enum.valueOf(xyz/changlu/Student, name);
    }
    private Student(String s, int i)
    {
        super(s, i);
    }
    public static final Student STUDENT;
    private static final Student $VALUES[];
    static 
    {
        STUDENT = new Student("STUDENT", 0);
        $VALUES = (new Student[] {
            STUDENT
        });
    }
}



测试程序:


检测IDEA中的私有无参构造创建实例:


class Test1{
    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        //通过Student的字节码类来获取构造器(空参)
        Constructor<Student> constructor = Student.class.getDeclaredConstructor();
        constructor.setAccessible(true);
        System.out.println(constructor.newInstance());
    }
}



通过检验我们能够知道IDEA中的反编译结果并不正确的!

IDEA中反编译出来的实际上有问题的,在枚举类中创建实例并不是通过无参构造来创建的,我们以jad工具反编译出的结果为准,是通过一个私有的有参构造来创建实例的。


class Test1{
    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        //获取单例
        System.out.println(Student.STUDENT);
        //传入参数获取无参构造
        Constructor<Student> constructor = Student.class.getDeclaredConstructor(String.class,int.class);
        constructor.setAccessible(true);
        System.out.println(constructor.newInstance());
    }
}



这个报错结果才应该出现的情况,与我们之前在newInstance()源码中抛出的异常结果相同!



总结


1、单例模式优点是减少内存系统开销,保证单个类只有一个实例;缺点是扩展困难,并发测试不利于调试,容易违背单一职责原则。


2、实现单例主要有三种方式:①饿汉式(线程安全的);②懒汉式(双重加锁+volatile);③内部静态类。通过反射技术上面三种方式都容易出现安全问题!


3、通过枚举类实现单例来解决反射带来的安全问题,因为在反射技术中newInstance()方法对于枚举类是无法进行实例化的!!!

相关文章
|
5月前
|
设计模式 算法 搜索推荐
Java 设计模式之策略模式:灵活切换算法的艺术
策略模式通过封装不同算法并实现灵活切换,将算法与使用解耦。以支付为例,微信、支付宝等支付方式作为独立策略,购物车根据选择调用对应支付逻辑,提升代码可维护性与扩展性,避免冗长条件判断,符合开闭原则。
1041 35
|
5月前
|
设计模式 消息中间件 传感器
Java 设计模式之观察者模式:构建松耦合的事件响应系统
观察者模式是Java中常用的行为型设计模式,用于构建松耦合的事件响应系统。当一个对象状态改变时,所有依赖它的观察者将自动收到通知并更新。该模式通过抽象耦合实现发布-订阅机制,广泛应用于GUI事件处理、消息通知、数据监控等场景,具有良好的可扩展性和维护性。
467 8
|
5月前
|
设计模式 网络协议 数据可视化
Java 设计模式之状态模式:让对象的行为随状态优雅变化
状态模式通过封装对象的状态,使行为随状态变化而改变。以订单为例,将待支付、已支付等状态独立成类,消除冗长条件判断,提升代码可维护性与扩展性,适用于状态多、转换复杂的场景。
752 157
|
5月前
|
设计模式 Java Spring
Java 设计模式之责任链模式:优雅处理请求的艺术
责任链模式通过构建处理者链,使请求沿链传递直至被处理,实现发送者与接收者的解耦。适用于审批流程、日志处理等多级处理场景,提升系统灵活性与可扩展性。
611 2
|
7月前
|
设计模式 缓存 Java
Java设计模式(二):观察者模式与装饰器模式
本文深入讲解观察者模式与装饰器模式的核心概念及实现方式,涵盖从基础理论到实战应用的全面内容。观察者模式实现对象间松耦合通信,适用于事件通知机制;装饰器模式通过组合方式动态扩展对象功能,避免子类爆炸。文章通过Java示例展示两者在GUI、IO流、Web中间件等场景的应用,并提供常见陷阱与面试高频问题解析,助你写出灵活、可维护的代码。
|
7月前
|
设计模式 安全 Java
Java设计模式(一):单例模式与工厂模式
本文详解单例模式与工厂模式的核心实现及应用,涵盖饿汉式、懒汉式、双重检查锁、工厂方法、抽象工厂等设计模式,并结合数据库连接池与支付系统实战案例,助你掌握设计模式精髓,提升代码专业性与可维护性。
|
7月前
|
设计模式 XML 安全
Java枚举(Enum)与设计模式应用
Java枚举不仅是类型安全的常量,还具备面向对象能力,可添加属性与方法,实现接口。通过枚举能优雅实现单例、策略、状态等设计模式,具备线程安全、序列化安全等特性,是编写高效、安全代码的利器。
|
安全 Java
Java常用的单例模式
Java常用的单例模式
204 0
|
安全 Java
Java常用的单例模式
版权声明:欢迎转载,请注明沉默王二原创。 https://blog.csdn.net/qing_gee/article/details/46608497 Java的单例single经常用到,但是使用枚举enum最佳。
1019 0
|
5月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
274 1