黑马程序员——JavaSE之对单例枚举和反射的看法一

本文探讨了如何利用反射破坏常见的单例模式实现,并介绍了使用枚举实现单例模式的优势。枚举单例不仅能够保证线程安全,还能有效防御反射攻击及防止序列化生成新实例。

                                          ------Java培训、Android培训、iOS培训、.Net培训、期待与您交流! -------



通常我们所使用的单例模式,我们都可以使用反射使它不再单例,如下饿汉式的单例模式:

public final class Singleton{
   private static final Singleton instance = new Singleton();
   private Singleton(){}
    
   public static Singleton getInstance(){
 	return instance;
   }
 
}
测试案例如下:
Singleton singleton1 = Singleton.getInstance();
Singleton singleton2 = Singleton.getInstance();

Constructor<Singleton> constructor = Singleton.class.getDeclaredConstructor();
constructor.setAccessible(true);
Singleton singleton3 = constructor.newInstance();

System.out.println(singleton1);
System.out.println(singleton2);
System.out.println(singleton3);
System.out.println(singleton1 == singleton2);
System.out.println(singleton2 == singleton3);

其中singleton1、singleton2都是通过我们所实现的单例模式来获取的对象,他们应该是同一个对象,singleton3则是通过反射获取无参构造器,constructor.setAccessible(true)来获取访问权限,最后通过无参构造器来创建一个对象singleton3,singleton3和上述两者应该不是同一个对象,测试结果如下:



说通常我们所使用的单例模式,我们都可以使用反射使它不再单例。然而单例使用枚举的话,却可以避免被反射


public enum Singleton {
    instance{

        @Override
        protected void read() {
            System.out.println("read");
        }

        @Override
        protected void write() {
            System.out.println("write");
        }

    };
    protected abstract void read();
    protected abstract void write();
}

以上是一个单例枚举的例子,而我们要获取该实例只需要Singleton.INSTANCE,并且此种方式可以保证该单例线程安全、防反射攻击、防止序列化生成新的实例。

枚举单例关于防反射攻击,当然和枚举的实现有关,枚举也是java类,我们对Singleton的class进行反编译,可以得到一个新的类



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);
        }
  }
	//如果此类含有ENUM修饰,调用该方法时会直接报错
	if((clazz.getModifiers() & Modifier.ENUM) != 0) 
		throw new IllegalArgumentException("Cannot reflectively create enum objects");
        ConstructorAccess ca = constructorAccessor;
        if(ca == null){
	    ca = acquireConstructorAccessor();	
	}
	return (T)ca.newInstance(initargs);
}

也就是说反射在通过newInstance创建对象时,会检查该类是否是枚举类,如果是,则抛出异常,反射失败。 也就是说使用枚举可以避免被反射,从而可以达到单例的效果。


可以发现以下特点:

  类的修饰abstract,所以没法实例化,反射也无能为力。
关于线程安全的保证,其实是通过类加载机制来保证的,我们看看INSTANCE的实例化时机,是在static块中,JVM加载类的过程显然是线程安全的。
对于防止反序列化生成新实例的问题还不是很明白,一般的方法我们会在该类中添加上如下方法,不过枚举中也没有显示的写明该方法。


### 黑马程序员 JavaSE 课程概述 #### 什么是JavaJava种广泛使用的面向对象编程语言,具有跨平台特性。它由Sun Microsystems于1995年推出,并迅速成为企业级应用开发的主要技术之[^1]。 #### 马程序员 JavaSE 基础教程的核心内容 黑马程序员JavaSE基础教程通常分为多个阶段,逐步深入讲解Java的基础知识高级特性。以下是该课程的些核心模块: 1. **Java入门** - 讲解Java的历史背景发展历程。 - 探讨Java的特点及其与其他编程语言的区别。 2. **集成开发环境(IDEA)** - 学习如何安装并配置IntelliJ IDEA或其他主流IDE。 - 使用IDE创建、运行调试简Java程序。 3. **基础语法** - 数据类型与变量声明。 - 运算符及表达式的使用方法。 - 控制结构如条件语句(if, switch)以及循环(for, while)。 4. **面向对象编程(OOP)** - 类与对象的概念介绍。 - 封装、继承多态三大特性的实现方式。 - 抽象类与接口的应用场景分析。 5. **异常处理机制** - 异常的基本概念及其分类。 - try-catch-finally块的作用范围说明。 - 自定义异常的设计原则。 6. **集合框架(Collection Framework)** - List、Set、Map等常用集合类型的比较选择依据。 - Iterator迭代器模式的工作原理详解。 - 并发环境下安全操作集合的方法探讨。 7. **输入输出流(I/O Streams)** - 文件读写基本流程演示。 - 字节流与字符流之间的区别联系解释。 - 缓冲区(Buffered Stream)提高效率的技术手段分享。 8. **多线程(Thread)** - 创建启动新线程的不同途径展示。 - 同步锁(Synchronized Block/Method)防止资源竞争现象发生的重要性强调。 - 线程间通信工具(Condition Variable,Pipe etc.)实际案剖析。 9. **网络编程(Network Programming)** - Socket套接字基础知识普及。 - 客户端服务器模型(Client-Server Model)构建过程指导。 - HTTP协议请求响应消息格式解析。 10. **数据库连接(JDBC)** - JDBC API主要组成部分功能描述。 - PreparedStatement预编译SQL语句提升性能优势阐述。 - ResultSet结果集遍历技巧总结。 ```java public class HelloWorld { public static void main(String[] args){ System.out.println("欢迎来到黑马程序员Java世界!"); } } ``` 以上仅为部分精选主题概览,具体学习路径还需参照官方文档或者购买正版教材获取更详尽的内容安排。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值