1. 什么是反射机制?
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制(注意关键词:运行状态)换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods。
反射机制主要提供的功能:
- 在运行时判断任意一个对象所属的类
- 在运行时构造任意一个类的对象
- 在运行时判断任意一个类所具有的成员变量和方法
- 在运行时调用任意一个对象的方法
2. java Reflection API简介
- Class类:代表一个类,位于java.lang包下
- Field类:代表类的成员变量(成员变量也称为类的属性)
- Method类:代表类的方法
- Constructor类:代表类的构造方法
- Array类:提供了动态创建数组,以及访问数组的元素的静态方法
3. java中的Class介绍
Class 类十分特殊,它没有共有的构造方法,被jvm调用的(简单的理解:new对象或者被类加载器加载的时候),在Java中,每个class都有一个相应的Class对象。也就是说,当我们编写一个类,编译完成后,在生成的.class文件中,就会产生一个Class对象,用于表示这个类的类型信息。
在运行期间,如果我们要产生某个类的对象,Java虚拟机(JVM)会检查该类型的Class对象是否已被加载。如果没有被加载,JVM会根据类的名称找到.class文件并加载它。一旦某个类型的Class对象已被加载到内存,就可以用它来产生该类型的所有对象。
class 对应的一些方法
Methon | Remark |
---|---|
getAnnotations | 获取这个类中所有注解(包含父类) |
getClassLoader | 获取加载这个类的类加载器 |
getDeclaredMethods | 获取这个类中的所有方法(不包括父类) |
getDeclaredMethods | 获取这个类中指定的方法(不包括父类) |
getMethods | 获取这个类中的所有方法(包括父类) |
getMethods | 获取这个类中指定的方法(包括父类) |
getDeclaredMethods | 获取这个类中的所有方法(不包括父类) |
getReturnType | 获取方法的返回类型 |
getParameterTypes | 获取方法的传入参数类型 |
isAnnotation | 测试这类是否是一个注解类 |
getDeclaredConstructors | 获取所有的构造方法 |
getSuperclass | 获取这个类的父类 |
getInterfaces | 获取这个类实现的所有接口 |
getFields | 获取这个类中所有被public修饰的成员变量 |
getField | 获取指定名字的被public修饰的成员变量 |
newInstance | 获取通过调用默认的(即无参数)构造函数创建的一个新实例 |
4. 反射中常用的方法
4.1 获取Class方法
//方式一,需要导入class对应的Package
Person person = new Person();
Class<? extends Person> personClazz01 = person.getClass();
//方式二,需要导入class对应的Package
//运用.class的方式来获取Class实例,对于基本数据类型的封装类,
//还可以采用.TYPE来获取相对应的
Class<? extends Person> personClazz03 = Person.class;
//方式二,不需要导入class对应的Package
try {
Class<?> personClazz02 = Class.forName("Person");
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
//方式四
ClassLoader classLoader = context.getClassLoader();
@SuppressWarnings("rawtypes")
Class SystemProperties = classLoader.loadClass("android.os.SystemProperties");
4.2 设置和获取私有成员变量
try {
Field field = person.getClass().getDeclaredField(fieldName);
// 参数值为true,打开禁用访问控制检查
//setAccessible(true) 并不是将方法的访问权限改成了public,而是取消java的权限控制检查。
//所以即使是public方法,其accessible 属相默认也是false
field.setAccessible(true);
field.set(person, "cyy");
return field.get(person);
} catch (Exception e) {
e.printStackTrace();
}
4.3 通过newInstance获取Class对象实例
Class<?> class1 = null;
class1 = Class.forName("com.android.reflect.Person");
//由于这里不能带参数,所以你要实例化的这个类Person,一定要有无参构造函数
Person person = (Person) class1.newInstance();
4.4 通过constructors 获取Class对象实例
class1 = Class.forName("com.android.reflect.Person");
//得到一系列构造函数集合
Constructor<?>[] constructors = class1.getConstructors();
try {
//默认构造函数
person1 = (Person) constructors[0].newInstance();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
//带参数的构造函数
person2 = (Person) constructors[1].newInstance(29, "zhuk");
4.5 通过Field操作Class对象成员变量(包括私有变量)
Class<?> class1 = null;
class1 = Class.forName("com.android.reflect.Person");
Object obj = class1.newInstance();
Field nameField = class1.getDeclaredField("name");
nameField.setAccessible(true);
nameField.set(obj, "cyy");
4.6 通过Method操作Class对象方法
Object instance = class1.newInstance();
//调用无参数方法fly
Method method = class1.getMethod("fly");
method.invoke(instance );
//调用带int参数smoke
method = class1.getMethod("smoke", int.class);
method.invoke(instance , 100);
通过反射我们可以修改父类的成员变量,解决了子类无法修改父类private成员变量的问题。
参考文章:
https://www.cnblogs.com/Free-Thinker/p/10384277.html