Java反射相关基础知识

引言

Java语言允许通过程序化的方式间接对Class进行操作。Class文件由类装载器装载
后,在JVM中将形成一份描述Class结构的元信息对象,通过该元信息对象可以获知Class的结构信息,如构造函数、属性和方法等。Java允许用户借由这个与Class相关的元信息对象间接调用Class对象的功能,这就为使用程序化方式操作Class对象开辟了途径。

简单实例

car类

public class Car {
	private String brand;
    private String color;
    private int maxSpeed;

    // 默认构造函数 和 带参构造函数
    public Car(){System.out.println("init car!!");}
    public Car(String brand,String color,int maxSpeed){
        this.brand = brand;
        this.color = color;
        this.maxSpeed = maxSpeed;
    }
    // 未带参的方法
    public void introduce() {
        System.out.println("brand:"+brand+";color:"+color+";maxSpeed:"+maxSpeed);
    }
    //省略get set方法、
}

ReflectTest

传统方式直接调用目标类的写法如Car car = new Car();car,setBrand("红旗CA72");Car car = new Car("红旗CA72", "黑色");,以下是通过Java反射机制以一种间接的方式操控目标类。

使用到的几个重要的反射类:ClassLoader、Class、Constructor、Method。通过这些反射类就可以直接调用目标Class的各项功能。

import java.lang.reflect.Constructor;
import java.lang.reflect.Method;

public class ReflectTest {
    public static Car initByDefaultConst()throws Throwable{
        //通过类装载器获取Car类对象
        ClassLoader loader = Thread.currentThread().getContextClassLoader();//获取当前线程的ClassLoader、
        Class clazz = loader.loadClass("com.smart.reflect.Car");//通过指定的全限定类名com.smart.reflect.Car装载Car类对应的反射实例、

        //获取类的默认构造器对象并通过它实例化Car
        Constructor cons = clazz.getDeclaredConstructor((Class[])null);
        Car car = (Car)cons.newInstance();//效果等同于new Car()、

        //通过反射方法设置属性
        Method setBrand = clazz.getMethod("setBrand",String.class);//方法名和方法入参的对象类型、
        setBrand.invoke(car,"红旗CA72");//调用目标类的方法,操作的目标类的对象实例,目标方法的入参、
        Method setColor = clazz.getMethod("setColor",String.class);
        setColor.invoke(car,"黑色");
        Method setMaxSpeed = clazz.getMethod("setMaxSpeed",int.class);
        setMaxSpeed.invoke(car,200);
        return car;
    }

    public static void main(String[] args) throws Throwable{
        Car car = initByDefaultConst();
        car.introduce();
    }
}

运行以上程序会打印 brand:红旗 CA72;color:黑色;maxSpeed:200

类装载器ClassLoader

类装载器的工作机制

类装载器就是寻找类的字节码文件并构造出类在JVM内部表示对象的组件。在Java中,类装载器把一个类装入JVM中,需要经过以下步骤:

  1. 装载:查找和导入 Class文件。

  2. 链接:执行校验、准备和解析步骤,其中解析步骤是可以选择的。

    ① 校验:检查载入 Class文件数据的正确性。
    ② 准备:给类的静态变量分配存储空间。
    ③ 解析:将符号引用转换成直接引用。

  3. 初始化:对类的静态变量、静态代码块执行初始化工作。

类装载工作由ClassLoader及其子类负责。ClassLoader是一个重要的Java运行时系统组件,它负责在运行时查找和装入 Class字节码文件。
JVM 在运行时会产生3个ClassLoader:根装载器、ExtClassLoader(扩展类装载器)、AppClassLoader(应用类装载器)。

根装载器:它使用C++语言编写,因而在Java中看不到它,根装载器负责装载JRE的核心类库,如JRE目标下的rt.jar、charsets.jar等。
ExtClassLoader:负责装载JRE扩展目录ext中的JAR类包;
AppClassLoader:负责装载Classpath路径下的类包。

ExtClassLoader和AppClassLoader 都是ClassLoader的子类,而根装载器不是。

这3个类装载器之间存在父子层级关系,即根装载器是ExtClassLoader的父装载器,ExtClassLoader是 AppClassLoader的父装载器。

在默认情况下,使用AppClassLoader装载应用程序的类。证明:

public class classLoaderTest {
    public static void main(string[] args){
        ClassLoader loader = Thread.currentThread().getContextClassLoader();
        System.out.println("current loader: " + loader);
        System.out.println("parent loader: " + loader.getParent());
        System.out.println("grandparent loader:" + loader.getParent().getParent());
    }
}

以上程序打印信息:

current loader:sun.misc.Launcher$AppClassLoader13171a
parent loader:sun.misc.Launcher$ExtClassLoader@15601ea
grandparent loader:null		//根装载器在Java中访问不到,所以返回null

JVM装载类时使用“全盘负责委托机制”。
全盘负责:当一个ClassLoader装载一个类时,除非显式地使用另一个ClassLoader,该类所依赖及引用的类也由这个ClassLoader载入;
委托机制:先委托父装载器寻找目标类,只有在找不到的情况下才从自己的类路径中查找并装载目标类。
这一点是从安全角度考虑的,如果有人编写了一个恶意的基础类(如java.lang.String)并装载到JVM中,后果将会很严重,但是由于有了“全盘负责委托机制”,java.lang.String永远是由根装载器来装载的,这样就避免了上述安全隐患的发生。

ClassLoader的重要方法

ClassLoader是一个抽象类,位于java.lang包中。下面是该类的一些重要接口方法

Class loadClass(String name):name参数指定类装载器需要装载类的名字,必须使用全限定类名,如com.smart. beans.Car。该方法有一个重载方法loadClass(String name,boolean resolve),resolve参数告诉类装载器是否需要解析该类。在初始化类之前,应考虑进行类解析的工作,但并不是所有的类都需要解析。如果JVM只需要知道该类是否存在或找出该类的超类,那么就不需要进行解析。

Class defineClass(String name,byte[] b, int off, int len):将类文件的字节数组转换成JVM内部的java.lang.Class对象。字节数组可以从本地文件系统、远程网络获取。参数name为字节数组对应的全限定类名。

Class findSystemClass(String name):从本地文件系统载入Class文件。如果本地文件系统不存在该Class文件,则将抛出ClassNotFoundException异常。该方法是JVM默认使用的装载机制。

Class findLoadedClass(String name):调用该方法来查看ClassLoader是否已装入某个类。如果已装入,那么返回java.lang.Class对象;否则返回null。如果强行装载已存在的类,那么将会抛出链接错误。

ClassLoader getParent():获取类装载器的父装载器。除根装载器外,所有的类装载器都有且仅有一个父装载器。ExtClassLoader的父装载器是根装载器,因为根装载器非Java语言编写,所以无法获得,将返回null。

类实例、类描述对象及类装载器的关系

除JVM默认的3个ClassLoader外,用户可以编写自己的第三方类装载器,以实现一些特殊的需求。类文件被装载并解析后,在JVM内将拥有一个对应的java.lang.Class类描述对象,该类的实例都拥有指向这个类描述对象的引用,而类描述对象又拥有指向关联 ClassLoader的引用,如图所示。
在这里插入图片描述
每个类在JVM中都拥有一个对应的java.lang.Class对象,它提供了类结构信息的描述。数组、枚举、注解及基本Java类型(如int、double等),甚至void 都拥有对应的Class对象。Class没有public的构造方法。Class对象是在装载类时由JVM通过调用类装载器中的defineClass()方法自动构造的。

Java反射机制

Class反射对象描述类语义结构,可以从Class对象中获取构造函数、成员变量、方法类等类元素的反射对象,并以编程的方式通过这些反射对象对目标类对象进行操作。这些反射对象类在java.reflect包中定义。

3个主要的反射类

Constructor:类的构造函数反射类,通过Class#getConstructors()方法可以获取类的所有构造函数反射对象数组。在Java 5.0中,还可以通过getConstructor(Class… parameterTypes)获取拥有特定入参的构造函数反射对象。Constructor的一个主要方法是newInstance(Object[] initargs),通过该方法可以创建一个对象类的实例,相当于new关键字。在Java 5.0中,该方法演化为更为灵活的形式: newInstance (Object…initargs)。

Method:类方法的反射类,通过Class#getDeclaredMethods()方法可以获取类的所有方法反射类对象数组Method[]。在Java 5.0中,可以通过getDeclaredMethod(String name, Class…parameterTypes)获取特定签名的方法,其中name为方法名;Class…为方法入参类型列表。Method最主要的方法是invoke(Object obj, Object[] args),其中obj表示操作的目标对象;args为方法入参。在Java 5.0中,该方法的形式调整为invoke(Object obj,Object… args)。此外,Method还有很多用于获取类方法更多信息的方法。

​ Class getReturnType():获取方法的返回值类型。
​ Class[] getParameterTypes():获取方法的入参类型数组。
​ Class[] getExceptionTypes():获取方法的异常类型数组。
​ Annotation[][] getParameterAnnotations():获取方法的注解信息,是Java 5.0中的新方法。

Field:类的成员变量的反射类,通过ClasstgetDeclaredFields()方法可以获取类的成员变量反射对象数组,通过ClasstgetDeclaredField(String name)则可以获取某个特定名称的成员变量反射对象。Field类最主要的方法是 set(Object obj, Object value),其中 obj表示操作的目标对象,通过value为目标对象的成员变量设置值。如果成员变量为基础类型,则用户可以使用Field类中提供的带类型名的值设置方法,如setBoolean(Object obj, boolean value)、setInt(Object obj, int value)等。

此外,Java还为包提供了Package反射类,在 Java 5.0中还为注解提供了AnnotatedElement反射类。总之,Java的反射体系保证了可以通过程序化的方式访问目标类中所有的元素。

通过反射调用private和protected成员变量和方法

对于private或protected成员变量和方法,只要JVM的安全机制允许,也可以通过反射进行调用。

PrivateCar

public class PrivateCar {
	// private成员变量;使用传统的类实例调用方式,只能在本类中访问
	private String color;
	// protected方法:使用传统的类实例调用方式,只能在子类和本包中访问
    protected void drive(){
        System.out.println("drive private car! the color is:"+color);
    }
}

color变量和drive()方法通过类实例变量无法在外部被访问和调用,但通过反射机制可以绕过这个限制。

ReflectTest

public class ReflectTest {
    public static void main(String[] args) throws Throwable{
        ClassLoader loader = Thread.currentThread().getContextClassLoader();
        Class clazz = loader.loadClass("com.smart.reflect.PrivateCar");
    
        PrivateCar pcar = (PrivateCar)clazz.newInstance();
        Field colorFld = clazz.getField("color");
    
        // 取消Java语言访问检查以访问private变量
        colorFld.setAccessible(true);
        colorFld.set(pcar,"红色");
    
        Method driveMtd = clazz.getDeclaredMethod("drive",(Class[])null);
    
        // 取消Java语言访问检查以访问protected方法
        driveMtd.setAccessible(true);
        driveMtd.invoke(pcar,(Class[])null);
    }   
}

在访问private和protected成员变量和方法时,必须通过setAccessible(boolean access)方法取消Java语言检查,否则将抛出IllegalAccessException。如果JVM的安全管理器设置了响应的安全机制,那么调用该方法将抛出SecurityException。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值