android进阶-java反射

1,反射的定义
一般情况下,我们使用某个类时必定知道它是什么类,是用来做什么的,并且能够获得此类的引用。于是我们直接对这个类进行实例化,之后使用这个类对象进行操作。

反射则是一开始并不知道我要初始化的类对象是什么,自然也无法使用 new 关键字来创建对象了。这时候,我们使用 JDK 提供的反射 API 进行反射调用。反射就是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;并且能改变它的属性。是Java被视为动态语言的关键。

Java反射机制主要提供了以下功能:

①在运行时构造任意一个类的对象

②在运行时获取或者修改任意一个类所具有的成员变量和方法

③在运行时调用任意一个对象的方法(属性)

也就是说我们并不知道怎么去new 对象就可以去用,如实际用法,这里我们不知道framework层中怎么去new serviceManager对象,但是我们想要去使用framework中的addService方法,并且这个方法还是有参数的。就可以用java的反射机制。

Object object = new Object();
Method addService;
addService = Class.forName("android.os.ServiceManager").getMethod(
        "addService", String.class, IBinder.class);//后面的两个参数用的是addservice默认参数列表
addService.invoke(object, SERVICE_NAME, new CAPermissionServiceManagerImp(new CAPermissionServiceManager())); 调用addservice方法,并传入实际参数
Log.e(TAG, "add to serviceManager success");

2,Class
反射始于Class,Class是一个类,封装了当前对象所对应的类的信息。

一个类中有属性,方法,构造器等,比如说 有一个Person类,一个Order类,一个Book类,这些都是不同的类,现在需要一个类,用来描述类,这就是 Class,它应该有类名,属性,方法,构造器等。Class是用来描述类的类。

Class类是一个对象照镜子的结果,对象可以看到自己有哪些属性,方法,构造器,实现了哪些接口等等。对于每 个类而言,JRE 都为其保留一个不变的 Class 类型的对象。一个 Class 对象包含了特定某个类的有关信息。 对象只 能由系统建立对象,一个类(而不是一个对象)在 JVM 中只会有一个Class实例。

2.1获取Class对象
获取Class对象的三种方式

①. 通过类名获取 类名.class

②. 通过对象获取 对象名.getClass()

③. 通过全类名获取 Class.forName(全类名) classLoader.loadClass(全类名)

例如获取People的Class对象:

public class People {
 
    private String name;
 
    private int age;
 
    public void setName(String name) {
        this.name = name;
    }
 
    public void setAge(int age) {
        this.age = age;
    }
 
    public String getName() {
        return name;
    }
 
    public int getAge() {
        return age;
    }
}
通过三种方式获取: 

//方式一 通过类名获取 类名.class
Class<People> peopleClass = People.class;
 
//方式二 通过对象获取 对象名.getClass()
People people =new People();
Class<? extends People> peopleClass1 = people.getClass();
 
//方式三 通过全类名获取 Class.forName(全类名) classLoader.loadClass(全类名)
try {
    Class<?> peopleClass2 = Class.forName("com.yuanzhen.People");
} catch (ClassNotFoundException e) {
    throw new RuntimeException(e);
}
方式一需要导入类的包,依赖太强,不导包就抛编译错误

方式二一般可以创建对象就不需要反射了

方式三最常用,没有依赖性

2.2判断是否为某个类的实例
//peopleClass 是否为People.class的实例
boolean assignableFrom = peopleClass.isAssignableFrom(People.class);
2.3创建实例
通过反射来生成对象主要有两种方式:

①使用Class对象的newInstance()方法来创建Class对象对应类的实例。

②先通过Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()方法来创建实例。这 种方法可以用指定的构造器构造类的实例。

例如:

//方式一 使用Class对象的newInstance()方法来创建Class对象对应类的实例
try {
    People people1 = peopleClass.newInstance();
} catch (IllegalAccessException e) {
    throw new RuntimeException(e);
} catch (InstantiationException e) {
    throw new RuntimeException(e);
}
//方式二 先通过Class对象获取指定的Constructor对象
//再调用Constructor对象的newInstance()方法来创建实例
//这种方法可以用指定的构造器构造类的实例
Constructor<People> constructor = null;
try {
    constructor = peopleClass.getConstructor(People.class);
    constructor.newInstance();
} catch (Exception e) {
    throw new RuntimeException(e);
}
3,构造器
上面提到了通过构造器可以创建对象,那么怎么获得构造器呢?

获得构造器的api:

Constructor getConstructor(Class[] params)    获得使用特殊的参数类型的public构造函数(包括父类)
Constructor[] getConstructors()    获得类的所有公共构造函数
Constructor getDeclaredConstructor(Class[] params)    获得使用特定参数类型的构造函数(包括私有)
Constructor[] getDeclaredConstructors()    获得类的所有构造函数(与接入级别无关)
例如:

public class People {
 
    public People(String name, int age) {
        this.name = name;
        this.age = age;
    }
    private People() {
 
    }
    public People(String name) {
        this.name = name;
    }
}
try {
    //获取Class对象
    Class<?> peopleClass = Class.forName("com.yuanzhen.People");
    //获得使用特殊的参数类型的public构造函数(包括父类)
    Constructor<?> constructor = peopleClass.getConstructor(String.class, int.class);
    //获得使用特定参数类型的构造函数(包括私有)
   constructor = peopleClass.getDeclaredConstructor();
   //获得类的所有公共构造函数
    Constructor<?>[] constructors = peopleClass.getConstructors();
    //获得类的所有构造函数(与接入级别无关)
    constructors = peopleClass.getDeclaredConstructors();
} catch (Exception e) {
    throw new RuntimeException(e);
}
4,成员变量
4.1获取成员变量
获取类的成员变量的api:

Field getField(String name)    获得命名的公共字段
Field[] getFields()    获得类的所有公共字段
Field getDeclaredField(String name)    获得类声明的命名的字段
Field[] getDeclaredFields()    获得类声明的所有字段
例如:

public class People {
 
    public String name;
 
    private int age;
 
    public void setName(String name) {
        this.name = name;
    }
 
    public void setAge(int age) {
        this.age = age;
    }
 
    public String getName() {
        return name;
    }
 
    public int getAge() {
        return age;
    }
}
try {
    //获取Class对象
    Class<?> peopleClass = Class.forName("com.yuanzhen.People");
    //获得命名的公共字段  name
    Field field = peopleClass.getField("name");
    //获得类声明的命名的字段(包括私有) age
    Field field1 = peopleClass.getDeclaredField("age");
    //获得类的所有公共字段
    Field[] fields = peopleClass.getFields();
    //获得类声明的所有字段
    Field[] fields1 = peopleClass.getDeclaredFields();
} catch (Exception e) {
    throw new RuntimeException(e);
}
4.2调用成员变量
将成员变量赋值:

try {
    //获取Class对象
    Class<?> peopleClass = Class.forName("com.yuanzhen.People");
    //获得命名的公共字段  name
    Field field = peopleClass.getField("name");
    //获得类声明的命名的字段(包括私有) age
    Field field1 = peopleClass.getDeclaredField("age");
    //创建对象
    Object people = peopleClass.newInstance();
    //暴力反射,解除私有限定
    field1.setAccessible(true);
    //将年龄设为20
    field1.set(people,20);
    //将姓名设为张三
    field.set(people,"张三");
} catch (Exception e) {
    throw new RuntimeException(e);
}
5,成员方法
5.1获取成员方法
获取方法信息的api:

Method getMethod(String name, Class[] params)    使用特定的参数类型,获得命名的公共方法
Method[] getMethods()    获得类的所有公共方法
Method getDeclaredMethod(String name, Class[] params)    使用特写的参数类型,获得类声明的命名的方法
Method[] getDeclaredMethods()    获得类声明的所有方法
例如:

public class People {
 
    public String name;
 
    private int age;
 
    private void setName(String name) {
        this.name = name;
    }
 
    public void setAge(int age) {
        this.age = age;
    }
 
    private String getName() {
        return name;
    }
 
    public int getAge() {
        return age;
    }
}
try {
    //获取Class对象
    Class<?> peopleClass = Class.forName("com.yuanzhen.People");
    //使用特定的参数类型,获得命名的公共方法
    Method method = peopleClass.getMethod("setAge", int.class);
    //使用特写的参数类型,获得类声明的命名的方法(包括私有)
    Method method1 = peopleClass.getDeclaredMethod("setName", String.class);
    //获得类的所有公共方法
    Method[] methods = peopleClass.getMethods();
    //获得类声明的所有方法(包括私有)
    Method[] methods1 = peopleClass.getDeclaredMethods();
    
} catch (Exception e) {
    throw new RuntimeException(e);
}
5.2调用成员方法
try {
    //获取Class对象
    Class<?> peopleClass = Class.forName("com.yuanzhen.People");
    //使用特定的参数类型,获得命名的公共方法
    Method method = peopleClass.getMethod("setAge", int.class);
    //使用特写的参数类型,获得类声明的命名的方法(包括私有)
    Method method1 = peopleClass.getDeclaredMethod("setName", String.class);
    //创建对象实例
    Object people = peopleClass.newInstance();
    //暴力反射,解除私有限定
    method.setAccessible(true);
    //调用方法setAge
    method.invoke(people,20);
    //调用方法 setName
    method1.invoke(people,"张三");
} catch (Exception e) {
    throw new RuntimeException(e);
}
6,总结
反射的常规使用就是这些,本文主要是对常用反射api的记录总结。
————————————————

                            版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
                        
原文链接:https://blog.csdn.net/y2653904/article/details/132795200

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值