Java编译和运行
Java程序在编译期间将.java文件编译成.class文件,然后在运行期间将.class文件添加到内存中。
原理
反射就是程序在运行状态,可以动态的获得类的信息并调用类的方法。
通过获得class对象,进而获得class对象所属类的属性和方法。
应用场景
一些框架使用反射实现
当我们编写程序时不知道使用的是哪个类,而是要根据运行时动态的判断是哪个类时,可以使用反射。
可以通过反射进行类的加载与实例化,这样,当类改变时,可直接通过配置文件修改类名,而不需要修改代码
比如:JDBC中的driver通过Class.forName()进行类的加载,这样当我的driver改变时,只需修改配置文件的driver名即可,不许修改代码。
为什么用class.forname
功能
得到一个对象所属的类
获得一个类的所有成员变量和方法
在运行时创建对象
在运行时调用对象的方法
缺点
- 反射调用比直接调用慢。我们能在每一版 JVM 的发布中看到关于反射 API 性能提升的改进,JIT 编译器的优化算法越来越好,但是反射方法的调用还是比直接调用要慢三倍以上。
- 类型安全,如果在代码中使用了方法引用,那么这只是对方法的引用而已。如果写了代码通过反射来调用方法并传错了参数,这个调用会在运行时失败,而并不是在编译或者加载时。
- 内部暴露,由于反射允许代码执行一些在正常情况下不被允许的操作(比如访问私有的属性和方法),所以使用反射可能会导致意料之外的副作用--代码有功能上的错误,降低可移植性。反射代码破坏了抽象性,因此当平台发生改变的时候,代码的行为就有可能也随着变化。
- 可追踪性 - 如果一个反射方法调用失败,找到导致问题的代码行可能会很棘手,因为 stack trace 通常很大。需要跟踪到很深地方查许多 invoke() 和 proxy() 调用。
代码测试
package reflect;
import java.lang.reflect.*;
class Test{
private int num = 1;
public int sum = 10;
public Object t;
public Test(){
System.out.println("public Constructor");
}
private Test(int i){
System.out.println("private Constructor");
}
public Test(char i){
System.out.println("public Constructor");
}
private void method(){
System.out.println("private method");
}
public int getNum(){
System.out.println("public method getNum");
return num;
}
public void getMethod(){
System.out.println("public method getMethod");
}
}
public class Reflect {
public static void main(String[] args){
try {
//加载类
System.out.println("*获得类对象*");<