Android反射机制

1. 什么是反射机制?
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制(注意关键词:运行状态)换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods。
反射机制主要提供的功能:

  1. 在运行时判断任意一个对象所属的类
  2. 在运行时构造任意一个类的对象
  3. 在运行时判断任意一个类所具有的成员变量和方法
  4. 在运行时调用任意一个对象的方法

2. java Reflection API简介

  1. Class类:代表一个类,位于java.lang包下
  2. Field类:代表类的成员变量(成员变量也称为类的属性)
  3. Method类:代表类的方法
  4. Constructor类:代表类的构造方法
  5. Array类:提供了动态创建数组,以及访问数组的元素的静态方法

3. java中的Class介绍

Class 类十分特殊,它没有共有的构造方法,被jvm调用的(简单的理解:new对象或者被类加载器加载的时候),在Java中,每个class都有一个相应的Class对象。也就是说,当我们编写一个类,编译完成后,在生成的.class文件中,就会产生一个Class对象,用于表示这个类的类型信息。

在运行期间,如果我们要产生某个类的对象,Java虚拟机(JVM)会检查该类型的Class对象是否已被加载。如果没有被加载,JVM会根据类的名称找到.class文件并加载它。一旦某个类型的Class对象已被加载到内存,就可以用它来产生该类型的所有对象。

class 对应的一些方法

MethonRemark
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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值