java中的反射

反射是什么?

java的反射是基于class对象也就是字节码文件生成实例对象的过程。由于字节码文件内记录这个对象的所有的方法,属性,因此可以通过class对象生成实例对象。

class对象内到底存了些什么?

1、属性

		class文件内存放代码内定义的所有属性,不管是什么修饰符修饰的都会记录下来。

2、方法

		class文件内存放代码内定义的所有方法,不管是什么修饰符修饰的都会记录下来。

3、常量池

	   常量池相当于class文件的资源库,存储了后面字段、接口、方法等需要的字符串,后面结构使用只需要指向常量池索引。

为什么会出现反射?

对于一个一般性程序来说,程序在代码书写完毕之后,程序就会按代码的书写去固定的运行。但是在很多情况下,我们需要动态的获取某些对象的某个属性,或者执行某些的对象的某些方法,使得程序更加的灵活,多变。

反射的实现原理

由于每一个对象都有自己对应的class对象,而这个class对象记录了这个对象所有的属性以及方法,那么我们完全可以通过class对象来拿到这个对象内的所有的属性和方法。

反射的使用

1、获取class对象

	clazz.getPackage().getName()//包名
	clazz.getSimpleName()//类名
	clazz.getName()//完整类名

2、通过class对象获取实例对象的属性

	getFields()//获取所有公开的成员变量,包括继承变量
	getDeclaredFields()//获取本类定义的成员变量,包括私有,但不包括继承的变量
	getField(变量名)
	getDeclaredField(变量名)

3、通过class对象获取实例对象的方法

	getMethods()//获取所有可见的方法,包括继承的方法
	getMethod(方法名,参数类型列表)
	getDeclaredMethods()//获取本类定义的的方法,包括私有,不包括继承的方法
	getDeclaredMethod(方法名,int.class,String.class)

4、反射出新的实例对象

	clazz.newInstance();//执行无参构造创建对象
	clazz.newInstance(666,”海绵宝宝”);//执行含参构造创建对象
	clazz.getConstructor(int.class,String.class)//获取构造方法

反射的注意点

性能不是很好,因为这些控制处理都不是固定的,对于JVM来说他需要先进行一定的处理才能知道该实例化出什么样的对象,然后进行处理。而且,jvm对于一些固定的代码是会有编译期,运行期优化的,但是动态代码来说就不能进行优化了。
安全问题,由于反射可以破坏任意修饰符带来的权限隔离,因此对于一些对程序理解不够深的同学来说,就可能在使用中出现不当操作,导致程序出现问题。
健壮性问题,由于代码是动态运行,那么运行时可能会出现的情形就可能是多样性的,如果没有预判到所有情形的处理,就可能对程序的健壮性造成一定的破坏。
### Java反射机制的原理与用法 #### 一、Java反射机制的基本原理 Java反射机制允许程序在运行时动态地访问类的信息,包括类的字段、方法和构造函数等。这种能力的核心在于`Class`类[^1]。每当一个类被加载到JVM中时,都会为其创建一个对应的`Class`对象,该对象包含了这个类的所有元信息。 获取`Class`对象的方式主要有三种: 1. 使用`.class`语法:例如 `String.class`。 2. 调用对象的`getClass()`方法:例如 `obj.getClass()`。 3. 利用`Class.forName(String className)`静态方法:例如 `Class.forName("java.lang.String")`[^3]。 一旦获得了`Class`对象,就可以通过它来获取类的各种组件,比如字段(`Field`)、方法(`Method`)和构造器(`Constructor`)。 --- #### 二、Java反射机制的主要功能 ##### 1. 动态创建对象 通过反射可以不依赖于传统的`new`关键字,在运行时动态创建对象。以下是实现方式的一个例子: ```java // 获取Class对象 Class<?> clazz = Class.forName("com.example.MyClass"); // 创建实例 Object instance = clazz.getDeclaredConstructor().newInstance(); ``` 这种方式特别适用于需要降低耦合度的场景,例如框架设计中的简单工厂模式优化[^3]。 ##### 2. 访问私有成员 即使某些字段或方法声明为`private`,也可以通过反射绕过访问控制修饰符进行访问。不过需要注意的是,默认情况下这些私有成员是不可见的,因此需要显式调用`setAccessible(true)`来改变其可访问状态[^2]。 下面是一个简单的演示代片段: ```java import java.lang.reflect.Field; public class ReflectionExample { private String secret = "This is a secret"; public static void main(String[] args) throws Exception { ReflectionExample obj = new ReflectionExample(); Field field = obj.getClass().getDeclaredField("secret"); field.setAccessible(true); // 设置为可访问 System.out.println(field.get(obj)); // 输出秘密值 } } ``` ##### 3. 调用任意方法 无论是公有还是私有的方法都可以借助反射技术完成调用操作。这里给出一段示例代用于说明这一过程: ```java import java.lang.reflect.Method; public class MethodReflection { private void sayHello(String name) { System.out.println("Hello, " + name); } public static void main(String[] args) throws Exception { MethodReflection mr = new MethodReflection(); Method method = mr.getClass().getDeclaredMethod("sayHello", String.class); method.setAccessible(true); method.invoke(mr, "World"); // 执行方法 } } ``` --- #### 三、性能考量 尽管反射非常强大,但它也伴随着一定的代价——主要是执行效率上的损失。这是因为反射打破了编译期间已知类型的约束,增加了额外的安全性和一致性检查工作量[^1]。所以在实际项目里应谨慎权衡利弊后再决定是否采用此手段解决问题。 --- #### 四、典型应用场景 - **框架开发**:像Spring这样的流行框架广泛运用了反射机制来进行IoC容器初始化等工作; - **调试工具**:许多现代IDE内置的功能(如断点跟踪)背后也是基于反射实现的; - **动态代理**:AOP编程模型下常用来增强目标业务逻辑的行为特性等等[^3]。 --- ### 结论 综上所述,虽然存在些许局限性,但毫无疑问Java反射是一项极其重要的技能点之一,掌握好它的理论基础及实践技巧无疑会对提升个人技术水平大有助益。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值