概述

JAVA 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为 java 语言的反射机制。

每个类都有一个 Class 对象,包含了与类有关的信息。当编译一个新类时,会产生一个同名的 .class 文件,该文件内容保存着 Class 对象。

类加载相当于 Class 对象的加载,类在第一次使用时才动态加载到 JVM 中。也可以使用 Class.forName("com.mysql.jdbc.Driver") 这种方式来控制类的加载,该方法会返回一个 Class 对象。

反射可以提供运行时的类信息,并且这个类可以在运行时才加载进来,甚至在编译时期该类的 .class 不存在也可以加载进来。

Class 和 java.lang.reflect 一起对反射提供了支持,java.lang.reflect 类库主要包含了以下三个类:

  • Field :可以使用 get() 和 set() 方法读取和修改 Field 对象关联的字段;
  • Method :可以使用 invoke() 方法调用与 Method 对象关联的方法;
  • Constructor :可以用 Constructor 的 newInstance() 创建新的对象。

反射的优点:

  • 可扩展性 :应用程序可以利用全限定名创建可扩展对象的实例,来使用来自外部的用户自定义类。
  • 类浏览器和可视化开发环境 :一个类浏览器需要可以枚举类的成员。可视化开发环境(如 IDE)可以从利用反射中可用的类型信息中受益,以帮助程序员编写正确的代码。
  • 调试器和测试工具 : 调试器需要能够检查一个类里的私有成员。测试工具可以利用反射来自动地调用类里定义的可被发现的 API 定义,以确保一组测试中有较高的代码覆盖率。

反射的缺点:

尽管反射非常强大,但也不能滥用。如果一个功能可以不用反射完成,那么最好就不用。在我们使用反射技术时,下面几条内容应该牢记于心。

  • 性能开销 :反射涉及了动态类型的解析,所以 JVM 无法对这些代码进行优化。因此,反射操作的效率要比那些非反射操作低得多。我们应该避免在经常被执行的代码或对性能要求很高的程序中使用反射。
  • 安全限制 :使用反射技术要求程序必须在一个没有安全限制的环境中运行。如果一个程序必须在有安全限制的环境中运行,如 Applet,那么这就是个问题了。
  • 内部暴露 :由于反射允许代码执行一些在正常情况下不被允许的操作(比如访问私有的属性和方法),所以使用反射可能会导致意料之外的副作用,这可能导致代码功能失调并破坏可移植性。反射代码破坏了抽象性,因此当平台发生改变的时候,代码的行为就有可能也随着变化。

获取Class对象的四种方式

如果我们动态获取到这些信息,我们需要依靠 Class 对象。Class 类对象将一个类的方法、变量等信息告诉运行的程序。Java 提供了四种方式获取 Class 对象:

  1. 知道具体类的时候:

    1
    Class alunbarClass = TargetObject.class;

    通过此方法获取Class对象不会初始化

  2. 通过Class.forName()传入类的路径获取:

    1
    Class alunbarClass1 = Class.forName("cn.javaguide.TargetObject");

    Class.forName(className)方法,内部实际调用的是一个native方法 forName0(className, true, ClassLoader.getClassLoader(caller), caller);

    第2个boolean参数表示类是否需要初始化,Class.forName(className)默认是需要初始化。

    一旦初始化,就会触发目标对象的 static块代码执行,static参数也会被再次初始化。

  3. 通过对象实例instance.getClass()获取:

    1
    2
    Employee e = new Employee();
    Class alunbarClass2 = e.getClass();
  4. 通过类加载器xxxClassLoader.loadClass()传入类路径获取

    1
    class clazz = ClassLoader.LoadClass("cn.javaguide.TargetObject");

    通过类加载器获取Class对象不会进行初始化,意味着不进行包括初始化等一些列步骤,静态块和静态对象不会得到执行

实例

  1. 创建一个反射操作的类TargerObject

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    package cn.javaguide;

    public class TargetObject {
    private String value;

    public TargetObject() {
    value = "JavaGuide";
    }

    public void publicMethod(String s) {
    System.out.println("I love " + s);
    }

    private void privateMethod() {
    System.out.println("value is " + value);
    }
    }
  2. 使用反射操作这个类的方法和参数

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    package cn.javaguide;

    import java.lang.reflect.Field;
    import java.lang.reflect.InvocationTargetException;
    import java.lang.reflect.Method;

    public class Main {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException, NoSuchFieldException {
    /**
    * 获取TargetObject类的Class对象并且创建TargetObject类实例
    */
    Class<?> tagetClass = Class.forName("cn.javaguide.TargetObject");
    TargetObject targetObject = (TargetObject) tagetClass.newInstance();
    /**
    * 获取所有类中所有定义的方法
    */
    Method[] methods = tagetClass.getDeclaredMethods();
    for (Method method : methods) {
    System.out.println(method.getName());
    }
    /**
    * 获取指定方法并调用
    */
    Method publicMethod = tagetClass.getDeclaredMethod("publicMethod",
    String.class);

    publicMethod.invoke(targetObject, "JavaGuide");
    /**
    * 获取指定参数并对参数进行修改
    */
    Field field = tagetClass.getDeclaredField("value");
    //为了对类中的参数进行修改我们取消安全检查
    field.setAccessible(true);
    field.set(targetObject, "JavaGuide");
    /**
    * 调用 private 方法
    */
    Method privateMethod = tagetClass.getDeclaredMethod("privateMethod");
    //为了调用private方法我们取消安全检查
    privateMethod.setAccessible(true);
    privateMethod.invoke(targetObject);
    }
    }

    输出内容:

    1
    2
    3
    4
    publicMethod
    privateMethod
    I love JavaGuide
    value is JavaGuide

静态编译和动态编译

  • 静态编译:在编译时确定类型,绑定对象
  • 动态编译:运行时确定类型,绑定对象

反射机制的优缺点

  • 优点:运行期类型的判断,动态加载类,提高代码灵活度。
  • 缺点:1,性能瓶颈:反射相当于一系列解释操作,通知 JVM 要做的事情,性能比直接的 java 代码要慢很多。2,安全问题,让我们可以动态操作改变类的属性同时也增加了类的安全隐患。

应用场景

在我们平时的项目开发过程中,基本上很少会直接使用到反射机制,但这不能说明反射机制没有用,实际上有很多设计、开发都与反射机制有关,例如模块化的开发,通过反射去调用对应的字节码;动态代理设计模式也采用了反射机制,还有我们日常使用的 Spring/Hibernate 等框架也大量使用到了反射机制。

举例:

  1. 我们在使用 JDBC 连接数据库时使用 Class.forName()通过反射加载数据库的驱动程序;
  2. Spring 框架的 IOC(动态加载管理 Bean)创建对象以及 AOP(动态代理)功能都和反射有联系;
  3. 动态配置实例的属性;
  4. ……