JavaSE 反射——反射概述及获取字节码的方式

一、反射概述

  • 动态获取信息以及动态调用对象方法的功能就称为反射机制,通过Java中的反射机制可以操作(读和修改)字节码文件(.class文件),让程序更加灵活,反射机制的相关类存在于java.lang.reflect.*包下
  • 几个重要的反射机制的相关类
    java.lang.Class:代表整个字节码(代表一个类型、整个类)
    java.lang.reflect.Method:代表字节码中的方法字节码(代表类中的方法)
    java.lang.reflect.Constructor:代表字节码中的构造方法字节码(代表类中的构造方法)
    java.lang.reflect.Field:代表字节码中的属性字节码(代表类中的成员变量(静态变量 + 实例变量))

二、获取字节码(java.lang.Class实例)的方式

1. Class.forName()

  • 装入类,并执行静态初始化工作(执行静态代码块),返回Class的对象
  • 语法格式
Class c = Class.forName("完整类名");
  • 静态方法,直接使用类名.方法名的方式
  • 方法的参数是一个字符串(完整类名),其中包名也不能省略
public class ReflectTest {
	public static void main(String[] args) {
		
		/**
		 * 通过Class.forName()获取Class
		 */
		try {
			// 编译时异常,try catch处理
			Class class1 = Class.forName("java.lang.System"); // 表示System.class文件,代表System类型
			Class class2 = Class.forName("java.util.Properties"); // 表示Properties.class文件,代表Properties类型
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}
	}
}
  • 如果只希望一个类的静态代码块执行,其它代码一律不执行,则可以使用 Class.forName(“完整类名”),因为这个方法的执行会导致类加载,将类加载到JVM中,而静态代码块是在类加载的时候执行
  • Class.forName()源码:
public static Class<?> forName(String className) throws ClassNotFoundException {
    Class<?> caller = Reflection.getCallerClass();
    return forName0(className, true, ClassLoader.getClassLoader(caller), caller);
}

private static native Class<?> forName0(String name, boolean initialize, ClassLoader loader, Class<?> caller)
  • initialize控制是否对类进行静态初始化,而Class.forName(String className)内部是true,所以会执行静态初始化工作(执行静态代码块)

2. getClass()

  • 语法格式
Class c = 对象.getClass();
  • getClass() 是Java中任何对象都有的方法
  • 即getClass() 是通过对象完成的,所以在创建对象时静态代码块和构造代码块都会执行,返回引用运行时真正所指的对象所属的类的Class的对象(因为子对象的引用可能会赋给父对象的引用变量中)
public class ReflectTest {
	public static void main(String[] args) {
		
		/**
		 * 通过getClass()获取Class
		 */
		HashMap<String, Integer> hashMap = new HashMap<>();
		Class c1 = hashMap.getClass();
		System.out.println(c1); // class java.util.HashMap
	}
}

3. .class属性

  • JVM将使用类装载器, 将类装入内存(前提是类还没有装入内存),不做类的静态初始化工作(不执行静态代码块),返回Class的对象
  • 语法格式
Class c = 任何类型.class;
  • Java中任何一种类型,包括基本数据类型,都有.class属性
public class ReflectTest {
	public static void main(String[] args) {
		
		/**
		 * 通过.class属性获取Class
		 */
		HashMap<String, Integer> hashMap = new HashMap<>();
		Class c1 = hashMap.getClass();
		Class c2 = HashMap.class;
		System.out.println(c1 == c2); // true
	}
}

三、反射的应用场景

  • Spring 实例化对象:当程序启动时,Spring 会读取配置文件applicationContext.xml并解析出里面所有的标签实例化到IOC容器中
  • 反射 + 工厂模式:通过反射消除工厂中的多个分支,如果需要生产新的类,无需关注工厂类,工厂类可以应对各种新增的类,反射可以使得程序更加健壮
  • JDBC连接数据库:使用JDBC连接数据库时,指定连接数据库的驱动类时用到反射加载驱动类

四、反射的缺点

1. 性能第一

  • 反射包括了一些动态类型,所以 JVM 无法对这些代码进行优化,因此,反射操作的效率要比那些非反射操作低得多,所以应该避免在经常被执行的代码或对性能要求很高的程序中使用反射

2. 安全限制

  • 使用反射技术要求程序必须在一个没有安全限制的环境中运行

3. 内部暴露

  • 由于反射允许代码执行一些在正常情况下不被允许的操作(比如访问私有的属性和方法),降低可移植性,另外反射代码破坏了抽象性,因此当平台发生改变的时候,代码的行为就有可能也随着变化
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Jayco江柯

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值