详解Java反射机制

共 6013字,需浏览 13分钟

 ·

2021-04-10 10:50

点击上方蓝色字体,选择“标星公众号”

优质文章,第一时间送达

反射机制的概念:

反射机制是指在程序运行过程中,对任意一个类都能够获取其所有属性和方法,并且对任意一个对象都能调用其任意一个方法。这种动态获取类和对象的信息,以及动态调用对象的方法的功能被称为Java语言的反射机制。


Class类的介绍:

说到反射,就不得不提起java.lang.Class这个类,JVM为每个加载的class创建了对应的Class实例,并在实例中保存了该class的所有信息,包括类名、包名、父类、实现的接口、所有方法、字段等。那么Class到底是个什么类呢?先看下它的部分源码:

public final class Class<T> implements java.io.Serializable,
                              GenericDeclaration,
                              Type,
                              AnnotatedElement {
    private Class(ClassLoader loader) {
        classLoader = loader;
    }
}

可以看出这个类被声明为final(不可变类),并且它的构造方法是private(私有的),即不可能通过new关键字来创建该实例,但是否可以通过反射机制(很强大)来创建那些构造方法为私有的类的实例呢?答案是几乎都可以,但Class这个类例外,下面可以来验证一下:

自定义一个构造方法是私有的类Test

class Test {
    private Test(){}
    // 只有Test的实例可调用
    public void sayHello(){
        System.out.println("hello");
    }
}

主类如下:

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

public class Main {
    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        // 获得Test类的class实例
        Class cTest = Test.class;
        // 获得Test类的构造方法
        Constructor constructor = cTest.getDeclaredConstructor();
        // 设置访问权限,对于执行非public的之前需先调用这个方法
        constructor.setAccessible(true);
        // 构造Test实例
        Test test = (Test) constructor.newInstance();
        // 调用sayHello方法
        test.sayHello();
    }
}


运行结果:


成功调用sayHello方法,即证明了我们通过反射可以实例化即使构造方法是私有的类的实例。


接下来再看看通过反射构造Class类的实例会怎样?

可以看到再代码14行执行setAccessible(true)方法时就抛异常了,前面已经说过,要想执行非public的必须得先调用setAccessible(true)方法。点进源码看看,

原来如此,它这里做了校验,你要是想访问Class类的构造方法,那么不好意思直接给你抛异常。Java虚拟机规范规定,所有的Class类的class实例均由JVM执行过程中动态加载,即在加载类时,读取定义该类的class文件(字节码文件)创建class实例并放在JVM的方法区中,我们是不可能创建的,只能获得。我们知道类是只会被JVM加载一次,这也说明了类的class实例只有一个,全局唯一且共享,所以不管使用什么方式获取某个类的class实例,该class实例是相同的,使用‘’==‘’返回true。


反射的应用:

Java中的对象有两种类型,即编译时类型和运行时类型。编译时类型在声明对象时所采用的类型,运行时类型指为对象赋值时所采用的类型。在如下代码中,person对象的编译时类型为Person,运行时类型为Student,

Person person = new Student();

因此,程序在编译期间无法预知该对象和类的真实信息,只能通过运行时信息来发现该对象和类的真实信息,而其真实信息(对象的属性和方法)通常通过反射机制来获取,这便是Java语言的反射机制的核心功能。


反射的API:

Java的反射API主要用于在运行过程中动态生成类、接口或对象等信息,其常用API如下:

  • Class类:用于获取类的属性、方法等信息。

  • Field类:表示类的成员变量,用于获取和设置类中的属性值

  • Method类:表示类的方法,用于获取方法的描述信息或执行某个方法

  • Constructor类:表示类的构造方法

反射的步骤:

1、获取想要操作的类的Class对象,该Class对象是反射的核心,通过它可以调用类的任意方法。

2、调用Class对象所对应的类中定义的方法,这是反射的使用阶段。

3、使用反射API来获取并调用类的属性和方法等信息。

获取Class对象的3种方法如下:

1、调用某个对象的getClass方法以获取该类对应的Class对象:

Test test = new Test();
Class c = test.getClass();

2、调用某个类的class属性以获取该类对应的Class对象:

Class c = Test.class;

3、调用Class类中的forName静态方法以获取该类对应的Class对象,这是最安全、性能也最好的方法:

Class c = Class.forName("test.Test");// 包路径加类名

访问字段:

Class类提供了以下几个方法来获取字段:

  • Field getField(name):根据字段名获取某个public的field(包括父类)

  • Field getDeclaredField(name):根据字段名获取当前类的某个field(不包括父类)

  • Field[] getFields():获取所有public的field(包括父类)

  • Field[] getDeclaredFields():获取当前类的所有field(不包括父类)


一个Field对象包含了一个字段的所有信息:

  • String getName():返回字段名称

  • Class getType():返回字段类型,也是一个Class实例

  • int getModifiers():返回字段的修饰符,它是一个int,不同的bit表示不同的含义

  • get(Object):返回指定对象Object的字段的值

  • set(Object, Object):设置字段的值,第一个Object参数是指定的实例,第二个Object参数是待修改的值

访问方法:

Class类提供了以下几个方法来获取Method:

  • Method getMethod(name, Class…):获取某个public的Method(包括父类)

  • Method getDeclaredMethod(name, Class…):获取当前类的某个Method(不包括父类)

  • Method[] getMethods():获取所有public的Method(包括父类)

  • Method[] getDeclaredMethods():获取当前类的所有Method(不包括父类)


一个Method对象包含一个方法的所有信息:

  • String getName():返回方法名称

  • Class getReturnType():返回方法返回值类型,也是一个Class实例

  • Class[] getParameterTypes():返回方法的参数类型,是一个Class数组

  • int getModifiers():返回方法的修饰符,它是一个int,不同的bit表示不同的含义

  • Object invoke(Object instance, Object… parameters):调用某个对象的方法


访问构造方法:

通过Class实例获取Constructor的方法如下:

  • getConstructor(Class…):获取某个public的Constructor

  • getDeclaredConstructor(Class…):获取某个Constructor

  • getConstructors():获取所有public的Constructor

  • getDeclaredConstructors():获取所有Constructor


创建对象的两种方式:

1、使用Class对象的newInstance方法创建该Class对象对应类的实例,这种方法要求该Class对象对应的类有默认的空构造器。

Class c = Test.class;
Test test = (Test) c.newInstance();

2、先使用Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance方法创建Class对象对应类的实例,通过这种方法可以选定构造方法创建实例。

// 获取构造方法Integer(int):
Constructor cons = Integer.class.getConstructor(int.class);
// 调用构造方法:
Integer n1 = (Integer) cons.newInstance(123);

注意:

调用非public的方法时,必须首先通过setAccessible(true)设置允许访问。setAccessible(true)可能会失败。


至此,Java的反射机制介绍到这里

————————————————

版权声明:本文为CSDN博主「b17a」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。

原文链接:

https://blog.csdn.net/weixin_45685353/article/details/115425373





粉丝福利:Java从入门到入土学习路线图

👇👇👇

👆长按上方微信二维码 2 秒


感谢点赞支持下哈 

浏览 30
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报