Java的反射机制

简介: Java 的反射机制允许在程序运行期间,借助反射 API 获取类的内部信息,并能直接操作对象的内部属性及方法。

介绍反射机制

Java 的反射机制允许在程序运行期间,借助反射 API 获取类的内部信息,并能直接操作对象的内部属性及方法。


Java 反射机制提供的功能:

  • 在运行时,使用反射分析类的能力,获取有关类的一切信息(类所在的包、类实现的接口、标注的注解、类的数据域、类的构造器、类的方法等)
  • 在运行时,使用反射分析对象,设置实例域的值,查看实例域的值。
  • 反射机制允许你调用任意方法(类的构造器方法、类的成员方法 等)

反射是一种功能强大且复杂的机制。使用反射机制的主要人员是工具构造者,而不是应用程序员。

Class 类

在程序运行期间,Java 运行时系统始终为所有的对象维护一个被称为运行时的类型标识。这个信息跟踪着每个对象所属的类。虚拟机利用运行时类型信息选择相应的方法执行。

然而,可以通过专门的 Java 类访问这些信息。保存这些信息的类被称为 Class。Object 类中的 getClass() 方法将会返回一个 Class 类型的实例。

如同用一个 Employee 对象表示一个特定的雇员属性一样,一个 Class 对象将表示一个特定类的属性。

虚拟机为每个类型管理一个 Class 对象。因此,可以利用 == 运算符实现两个 Class 对象比较的操作。

// 获得 Class 对象的多种方式:
public static void main(String[] args) {
    // 方式 1
    // 如果 T 是任意的 Java 类型 (或 void 关键字), T.class 将代表匹配的 Class 对象。
    Class<Person> clazz1 = Person.class;

    // 方式 2
    Person person = new Person();
    Class clazz2 = person.getClass();

    // 方式 3
    try {
        Class clazz3 = Class.forName("类的路径");
    } catch (ClassNotFoundException e) {
        throw new RuntimeException(e);
    }

    // 方式4
    // 获取到 ClassLoader(这里获取到的是:AppClassLoader)
    ClassLoader classLoader = ClassLoader.getSystemClassLoader();
    try {
        Class clazz4 = classLoader.loadClass("类的路径");
    } catch (ClassNotFoundException e) {
        throw new RuntimeException(e);
    }
}

还有一个很有用的方法:Class 类的 newlnstance(),可以用这个方法来动态地创建一个类的实例。newlnstance() 方法调用默认的构造器(没有参数的构造器)初始化新创建的对象。如果这个类没有默认的构造器,就会抛出一个 InstantiationException 异常。

将 Class 类的 forName() 方法与 Class 类的 newlnstance() 方法配合起来使用,可以根据存储在字符串中的类名创建一个对象。

public static void main(String[] args) throws Exception {
    String className = "java.util.Random";
    Object object = Class.forName(className).newInstance();
}

如果需要以这种方式向希望按名称创建的类的构造器提供参数,就不要使用上面那条语句,而必须使用 Constructor 类中的 newlnstance() 方法。

分析类的能力

在运行时,使用反射分析类的能力。

下面简要地介绍一下反射机制最重要的内容:检查类的结构。在 java.lang.reflect 包中有三个类 Field、Method 和 Constructor 分别用于描述类的数据域、类的方法和类的构造器。


这三个类都有一个叫做 getName() 的方法,用来返回项目的名称。

Field 类有一个 getType() 方法,用来返回描述数据域所属类型的 Class 对象。

Method 类和 Constructor 类有能够报告参数类型的方法,Method 类还有一个可以报告返回类型的方法。

这三个类还有一个叫做 getModifiers() 的方法,它将返回一个整型数值,用不同的位开关描述 public 和 static 这样的修饰符使用状况。另外, 还可以利用 java.lang.reflect 包中的 Modifier 类的静态方法分析 getModifiers() 返回的整型数值。例如,可以使用 Modifier 类中的 isPublic()、isPrivate() 或 isFinal() 判断方法或构造器是否是 public、private 或 final 的。我们需要做的全部工作就是调用 Modifier 类的相应方法,并对返回的整型数值进行分析,另外,还可以利用 Modifier.toString() 方法将修饰符打印出来。


Class 类的 getFields()、getMethods() 和 getConstructors() 方法将分别返回类中声明的 public 域、public 方法和 public 构造器数组,其中包括父类的公有成员。

Class 类的 getDeclareFields()、getDeclareMethods() 和 getDeclaredConstructors() 方法将分别返回类中声明的全部的数据域、全部的方法和全部的构造器,其中包括私有和受保护成员,但不包括父类的成员。

分析对象

在运行时,使用反射分析对象。

从前面一节中,已经知道如何查看任意对象的数据域的名称和类型:

  • 获得对应的 Class 对象。
  • 调用 Class 对象的 getDeclaredFields() 方法。

本节将进一步查看数据域的实际内容。当然,在编写程序时,如果知道想要査看的数据域的名称和类型,查看指定的数据域是一件很容易的事情。而利用反射机制可以查看在编译时还不清楚的数据域。

查看数据域值的关键方法是 Field 类中的 get() 方法。如果 f 是一个 Field 类型的对象(例如,通过 getDeclaredFields() 得到的对象),obj 是某个包含 f 域的类的对象,f.get(obj) 将返回一个对象,其值为 obj 对象的 f 域的当前值。

当然,可以获得就可以设置。调用 f.set(obj, value) 可以将 obj 对象的 f 域设置成新值。


public static void main(String[] args) {
    Employee harry = new Employee("Harry Hacker", 35000, 10, 1, 1989);
    Class cl = harry.getClass();
    
    // the class object representing Employee
    Field f = cl.getDeclaredField("name");
    // the name field of the Employee class
    Object v = f.get(harry);
    // the value of the name field of the harry object, i .e., the String object "Harry Hacker"
}

实际上,上面这段代码存在一个问题。由于 name 是一个私有域,所以 get() 方法将会抛出一个 illegalAccessException。只有利用 get() 方法才能得到可访问域的值。除非拥有访问权限,否则 Java 安全机制只允许査看任意对象有哪些域,而不允许读取它们的值。

反射机制的默认行为受限于 Java 的访问控制。然而,如果一个 Java 程序没有受到安全管理器的控制,就可以覆盖访问控制。为了达到这个目的,需要调用 Field、Method 或 Constructor 对象的 setAccessible() 方法。例如:

f.setAtcessible(true); // now OK to call f.get(harry);

setAccessible() 方法是 AccessibleObject 类中的一个方法,AccessibleObject 类是 Field、Method 和 Constructor 类的公共父类。这个特性是为调试、持久存储和相似机制提供的。

调用任意方法

在 C 和 C++ 中,可以从函数指针执行任意函数。从表面上看,Java 没有提供方法指针,即将一个方法的存储地址传给另外一个方法,以便第二个方法能够随后调用它。事实上,Java 的设计者曾说过:方法指针是很危险的,并且常常会带来隐患。他们认为 Java 提供的接口(interface)是一种更好的解决方案。然而,反射机制允许你调用任意方法。

为了能够看到方法指针的工作过程,先回忆一下利用 Field 类的 get() 方法查看数据域值的过程。与之类似,在 Method 类中有一个 invoke() 方法,它允许调用包装在当前 Method 对象中的方法。


可以使用 method 对象实现 C 语言中函数指针(或 C# 中的委派)的所有操作。同 C 一样,这种程序设计风格并不太简便,出错的可能性也比较大。如果在调用方法的时候提供了一个错误的参数,那么 invoke() 方法将会抛出一个异常。

另外, invoke() 方法的参数和返回值必须是 Object 类型的。这就意味着必须进行多次的类型转换。这样做将会使编译器错过检查代码的机会。因此,等到测试阶段才会发现这些错误,找到并改正它们将会更加困难。

在进行类型转换的过程中,编译器无法检查代码中类型转换的正确性,也就是无法保证转换后的类型与原始类型是兼容的。这样就会增加程序出错的可能性,并且如果出现错误的话,调试和修正也会更加困难。

不仅如此,使用反射获得方法指针的代码执行要比直接调用方法明显慢一些。

有鉴于此,建议仅在必要的时候才使用 Method 对象,而最好使用接口以及 Java8 中的 lambda 表达式。

特别要重申:建议 Java 开发者不要使用 Method 对象的回调功能。使用接口进行回调会使得代码的执行速度更快,更易于维护。

参考资料

《Java核心技术卷一:基础知识》(第10版)第 5 章:继承 5.7 反射

相关文章
|
5月前
|
设计模式 人工智能 安全
AQS:Java 中悲观锁的底层实现机制
AQS(AbstractQueuedSynchronizer)是Java并发包中实现同步组件的基础工具,支持锁(如ReentrantLock、ReadWriteLock)和线程同步工具类(如CountDownLatch、Semaphore)等。Doug Lea设计AQS旨在抽象基础同步操作,简化同步组件构建。 使用AQS需实现`tryAcquire(int arg)`和`tryRelease(int arg)`方法以获取和释放资源,共享模式还需实现`tryAcquireShared(int arg)`和`tryReleaseShared(int arg)`。
249 32
AQS:Java 中悲观锁的底层实现机制
|
3月前
|
人工智能 缓存 安全
Java中的反射机制:深入探索与应用
Java反射机制是程序运行时动态获取类信息并操作类成员的特性,具备高度灵活性,但也伴随性能与安全风险。本文详解反射的基本用法、高级应用及最佳实践,助你掌握这一强大工具的正确使用方式。
136 0
|
5月前
|
人工智能 Java 关系型数据库
Java——SPI机制详解
SPI(Service Provider Interface)是JDK内置的服务提供发现机制,主要用于框架扩展和组件替换。通过在`META-INF/services/`目录下定义接口实现类文件,Java程序可利用`ServiceLoader`动态加载服务实现。SPI核心思想是解耦,允许不同厂商为同一接口提供多种实现,如`java.sql.Driver`的MySQL与PostgreSQL实现。然而,SPI存在缺陷:需遍历所有实现并实例化,可能造成资源浪费;获取实现类方式不够灵活;多线程使用时存在安全问题。尽管如此,SPI仍是Java生态系统中实现插件化和模块化设计的重要工具。
159 0
|
3月前
|
人工智能 前端开发 安全
Java开发不可不知的秘密:类加载器实现机制
类加载器是Java中负责动态加载类到JVM的组件,理解其工作原理对开发复杂应用至关重要。本文详解类加载过程、双亲委派模型及常见类加载器,并介绍自定义类加载器的实现与应用场景。
186 4
|
3月前
|
人工智能 安全 Java
掌握Java反射:在项目中高效应用反射机制
Java反射是一种强大功能,允许程序在运行时动态获取类信息、创建对象、调用方法和访问字段,提升程序灵活性。它在框架开发、动态代理、注解处理等场景中广泛应用,如Spring和Hibernate。但反射也存在性能开销、安全风险和代码复杂性,应谨慎使用。
|
4月前
|
人工智能 Java
Java中的反射机制:深入探索与应用
本文介绍了Java反射机制的基本概念、用途及其实现方式。反射机制允许程序在运行时动态获取类的属性和方法,并调用它们,适用于处理私有成员或权限受限的情况。文章详细讲解了`Class`类的功能,包括获取类的方法、属性、注解、构造器等信息,以及通过四种方式获取`Class`对象的示例代码。此外,还探讨了类加载器、继承关系判断、动态代理等高级内容,展示了如何在运行时创建接口实例并处理方法调用。文末提供了完整的代码示例以加深理解。
Java中的反射机制:深入探索与应用
|
5月前
|
Java 区块链 网络架构
酷阿鲸森林农场:Java 区块链系统中的 P2P 区块同步与节点自动加入机制
本文介绍了基于 Java 的去中心化区块链电商系统设计与实现,重点探讨了 P2P 网络在酷阿鲸森林农场项目中的应用。通过节点自动发现、区块广播同步及链校验功能,系统实现了无需中心服务器的点对点网络架构。文章详细解析了核心代码逻辑,包括 P2P 服务端监听、客户端广播新区块及节点列表自动获取等环节,并提出了消息签名验证、WebSocket 替代 Socket 等优化方向。该系统不仅适用于农业电商,还可扩展至教育、物流等领域,构建可信数据链条。
|
5月前
|
人工智能 JavaScript Java
Java反射机制及原理
本文介绍了Java反射机制的基本概念、使用方法及其原理。反射在实际项目中比代理更常用,掌握它可以提升编程能力并理解框架设计原理。文章详细讲解了获取Class对象的四种方式:对象.getClass()、类.class、Class.forName()和类加载器.loadClass(),并分析了Class.forName()与ClassLoader的区别。此外,还探讨了通过Class对象进行实例化、获取方法和字段等操作的具体实现。最后从JVM类加载机制角度解析了Class对象的本质及其与类和实例的关系,帮助读者深入理解Java反射的工作原理。
104 0
|
7月前
|
缓存 Dubbo Java
理解的Java中SPI机制
本文深入解析了JDK提供的Java SPI(Service Provider Interface)机制,这是一种基于接口编程、策略模式与配置文件组合实现的动态加载机制,核心在于解耦。文章通过具体示例介绍了SPI的使用方法,包括定义接口、创建配置文件及加载实现类的过程,并分析了其原理与优缺点。SPI适用于框架扩展或替换场景,如JDBC驱动加载、SLF4J日志实现等,但存在加载效率低和线程安全问题。
278 7
理解的Java中SPI机制
|
6月前
|
存储 Java 编译器
Java 中 .length 的使用方法:深入理解 Java 数据结构中的长度获取机制
本文深入解析了 Java 中 `.length` 的使用方法及其在不同数据结构中的应用。对于数组,通过 `.length` 属性获取元素数量;字符串则使用 `.length()` 方法计算字符数;集合类如 `ArrayList` 采用 `.size()` 方法统计元素个数。此外,基本数据类型和包装类不支持长度属性。掌握这些区别,有助于开发者避免常见错误,提升代码质量。
473 1