目录
- 【Android】注解框架(一)-- 基础知识Java 反射
- 【Android】注解框架(二)-- 基础知识(Java注解)& 运行时注解框架
- 【Android】注解框架(三)-- 编译时注解,手写ButterKnife
- 【Android】注解框架(四)-- 一行代码注入微信支付
定义
主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。
反射机制就是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
用一句话总结就是反射可以实现在运行时可以知道任意一个类的属性和方法。
作用
- 指的是可以于运行时加载,探知和使用编译期间完全未知的类
- 程序在运行状态中, 可以动态加载一个只有名称的类, 对于任意一个已经加载的类,都能够知道这个类的所有属性和方法; 对于任意一个对象,都能调用他的任意一个方法和属性
- 加载完类之后, 在堆内存中会产生一个Class类型的对象(一个类只有一个Class对象), 这个对象包含了完整的类的结构信息,而且这个Class对象就像一面镜子,透过这个镜子看到类的结构
Class对象
Class是反射的基础,是java.lang.Class
的实例对象,并且Class是所有类的类类型。
private Class(ClassLoader loader) {
// Initialize final field for classLoader. The initialization value of non-null
// prevents future JIT optimizations from assuming this final field is null.
classLoader = loader;
}
通过源码可以看到Class的构造函数是私有的,只有JVM才可以创建Class对象,我们不能通过new Class()
的方式获取到Class对象。
每个类被加载进入内存之后,系统就会为该类生成一个对应的java.lang.Class对象,通过该Class对象就可以访问到JVM中的这个类.
假设有Person
类,我们可以通过以下的方法来获取到Class:
- 通过Person的对象,person.getClass()
- Person.class
- Class.forName(Person的全路径包名)
获取到了Class之后我们就能通过Java提供的API来访问或者修改Person类的各种属性和方法。
创建对象
我们在获取到了Class之后可以来创建对象,最常见的做法:
Class personClazz = Class.forName(com.example.Person);
Object person = personClazz.newInstance();
通过Class我们只能通过无参构造函数来创建对象。
另外我们也可以通过Class获取到构造函数之后通过构造函数来创建对象。
获取构造函数
// 获得该类所有的构造器,不包括其父类的构造器
public Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes)
// 获得该类所以public构造器,包括父类
public Constructor<T> getConstructor(Class<?>... parameterTypes)
我们可以通过获取构造函数之后来创建对象,举个例子:
public class Person{
private String name;
public Person(String name){
this.name = name;
}
}
public class Test{
public static void main(String[] args){
Class clazz = Class.forName("com.example.Person");
Constructor constructor = clazz.getDeclaredConstructor(String.class);
constructor.newInstance("张三");
}
}
获取方法
// 得到该类所有的方法,不包括父类的
public Method getDeclaredMethod(String name, Class<?>... parameterTypes)
// 得到该类所有的public方法,包括父类的
public Method getMethod(String name, Class<?>... parameterTypes)
// 得到该类所有的方法,不包括父类的或者
Method[] methods = c.getDeclaredMethods();
// 得到该类所有的public方法,包括父类的
Method[] methods = c.getMethods();
获取单个方法的Class<?>... parameterTypes
参数表示的是想反射的方法的参数的类类型。
举个例子:
public class Person{
private String name;
public void setName(String name){
this.name = name;
}
}
public class Test{
public static void main(String[] args){
Class clazz = Class.forName("com.example.Person");
Object person = clazz.newInstance();
Method method = clazz.getDeclaredMethod("setName", String.class);
method.invoke(person, "张三");
}
}
最后通过invoke方法调用方法,其中第一个参数是实例对象,后面的是参数。
获取属性
// 获得该类自身声明的所有变量,不包括其父类的变量
public Field getDeclaredField(String name)
// 获得该类自所有的public成员变量,包括其父类变量
public Field getField(String name)
// 返回 Field 对象的一个数组
public Field[] getDeclaredFields()
其他
// 将此对象的 accessible 标志设置为指示的布尔值。值为 true 则指示反射的对象在使用时应该取消 Java 语言访问检查。值为 false 则指示反射的对象应该实施 Java 语言访问检查。
void setAccessible(boolean flag)
// 获取属性或者方法或者类上的注解
<T extends Annotation> T getAnnotation(Class<T> annotationClass)
Annotation[] getAnnotations()
其他具体方法和属性可以参考:Java api 链接,搜索java.lang.reflect包。