java进阶之反射
1.1、什么时反射机制?反射机制有什么用?
- 反射机制:可以操作字节码文件
- 作用:可以让程序更加灵活
1.2、反射的应用:
Java中的对象有两种类型,即编译时类型和运行时类型。编译时类型在声明对象时所采用的类型,运行时类型指为对象赋值时所采用的类型。在如下代码中,person对象的编译时类型为Person,运行时类型为Student,
Person person = new Student();
因此,程序在编译期间无法预知该对象和类的真实信息,只能通过运行时信息来发现该对象和类的真实信息,而其真实信息(对象的属性和方法)通常通过反射机制来获取,这便是Java语言的反射机制的核心功能。
1.3、反射的API:
Java的反射API主要用于在运行过程中动态生成类、接口或对象等信息,其常用API如下:
- Class类:用于获取类的属性、方法等信息。
- Field类:表示类的成员变量,用于获取和设置类中的属性值
- Method类:表示类的方法,用于获取方法的描述信息或执行某个方法
- Constructor类:表示类的构造方法
1.3、反射的步骤:
- 获取想要操作的类的Class对象,该Class对象是反射的核心,通过它可以调用类的任意方法。
- 调用Class对象所对应的类中定义的方法,这是反射的使用阶段。
- 使用反射API来获取并调用类的属性和方法等信息。
获取Class对象的3种方法如下:
- 调用某个对象的getClass方法以获取该类对应的Class对象:
Test test = new Test();
Class c = test.getClass();
- 调用某个类的class属性以获取该类对应的Class对象:
Class c = Test.class;
- 调用Class类中的forName静态方法以获取该类对应的Class对象,这是最安全、性能也最好的方法:
Class c = Class.forName("test.Test");// 包路径加类名
1.4访问字段:
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参数是待修改的值
1.5、访问方法:
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):调用某个对象的方法
1.6、访问构造方法:
通过Class实例获取Constructor的方法如下:
- getConstructor(Class…):获取某个public的Constructor
- getDeclaredConstructor(Class…):获取某个Constructor
- getConstructors():获取所有public的Constructor
- getDeclaredConstructors():获取所有Constructor
创建对象的两种方式:
- 使用Class对象的newInstance方法创建该Class对象对应类的实例,这种方法要求该Class对象对应的类有默认的空构造器。
Class c = Test.class;
Test test = (Test) c.newInstance();
- 先使用Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance方法创建Class对象对应类的实例,通过这种方法可以选定构造方法创建实例。
// 获取构造方法Integer(int):
Constructor cons = Integer.class.getConstructor(int.class);
// 调用构造方法:
Integer n1 = (Integer) cons.newInstance(123);
1.7、 关于路径问题
String path=Thread.currentThread().getContextClassLoader().getResource("src下的相对路径").getPath();
这种方式是为了获取一个文件的绝对路径。
但是该文件要求放在类路径下,换句话说:就是放在src下。
注意:
调用非public的方法时,必须首先通过setAccessible(true)设置允许访问。setAccessible(true)可能会失败。