-----------android培训、java培训、java学习型技术博客、期待与您交流!------------
一、反射的基石
根据java的面向对象的思想,把事物都用类来进行了描述。那么问题来了,“类”它也是一种事物,我们是不是也要单独用一个类来描述“类”这种事物呢?
java中就给我们提供了这么样一个类,专门用来描述“类”这种事物。这个类就是Class类。
那么一个类,有什么样的共性内容进行抽取呢?比如:类名,构造函数,成员变量,成员函数等。
Class类是没有构造函数的,也就是说它不能new对象,因为不能初始化。
那么我们怎么获得这个类的对象呢,可以通过字节码文件来完成。
我们知道每个类在编译的时候都会产生一个扩展名是" .class "的文件,这个文件就是字节码文件。
1.得到字节码方式有三种:
(1) 类名.class (2)对象.getClass(); (3)Class.forName("类的全名");
做反射,主要用第三种。
2.九个预定义的.calss实例对象: 包含8个基本数据类型和void
isPrimitive(); 该方法是用来判断是不是基本数据类型字节码
总之,只要在源程序中出现的类型,都有各自的Class实例对象。
二、理解反射
1.什么是反射?
java反射机制是在运行状态中,对于任意一个类(class文件),都能够知道这个类的所有属性和方法,对于任意一个对象,都能够调用它的任意一个方法和属性,这种动态获取信息以及动态调用对象的方法的功能,就称之为反射机制。
简单的说,反射就是把java类中的各种成分映射成相应的java类。
比如: 构造函数 constructor ---------->Constructor类
成员变量 field ---------> Field类
成员方法 method --------->Method类
三、通过反射创建对象的方式
这里主要是获取构造函数。
构造函数 constructor ---------->Constructor类
第一种方式:寻找该名称类文件,并加载进内存,并产生Class 对象
代码例:Class c = Class.forName("Person"); // 通过类名获取其字节码文件对象
然后使用Class类中newInstance();方法来产生该类对象。
Object obj = c.newInstance(); //其实调用空参数构造函数,初始化。
但是第一种方式有个局限,如果原类中没有空参构造函数,或者构造函数时私有的,就会报异常。当要获取指定名称类中的所体现的对象时,而该对象初始化不使用空参怎么办?解决办法就是通过反射,获取该类中的构造函数,然后通过构造函数来创建该类对象。
第二种方式:
首先获得该类的构造函数。
Class class = Class.forName("Person");
通过getConstructor();方法来获取其指定构造函数
Constructor cons = class.getConstructor(Sting.class,int.class); // 意思获取参数列表是(String , int )这个的构造函数。
然后通过构造器对象创建Person类对象。
Object obj = cons.getInstance("xiaoming",20); // 创建对象,并初始化
四、获取类中的成员变量
成员变量 field ---------> Field类
获取指定类中成员变量步骤:
(1)获取指定对象的字节码文件。
(2)通过字节码文件拿到成员变量,通过getField(参数名); 来获取公有成员变量。
如果想获取类中所有成员变量,通过 getDeclaredField() ,包括私有也可以。
代码例:比如现在有个类 Demo ,该类中有两个 公有成员变量 int x , int y
Class class = Class.forName("Demo");
Field fieldY = class.getField("y");
这里的fieldY 不是对象身上的变量,而是类上的,要用对应的它去取某个对象上对应的值。比如通过类中的get 方法。
Constructor cons = class.getConstructor(int.class,int.class); 获取构造函数
Demo d = cons.newInstance(3,5); 建立Demo的对象
int y = fieldY.get(d); 获取d对象上的y值;
注意:如果成员变量是私有的,那么即使拿到也不能通过一般的方式访问到。只能通过暴力反射。
暴力反射: setAccessible(true); 取消权限检查。
五、获取指定公有函数
成员方法 method --------->Method类
首先获得指定了的字节码文件,然后用该字节码文件对象去获取这个类中的方法。getMethod(方法名,参数列表) ;来获取指定的方法。
比如:String str = "abc"; 我想获取指定位置上的字符。那么想起String类中提供了charAt(int x ) 方法可以实现我们的想法。
那么如何利用反射的原理来做呢?
Method methodCharAt = Sting.class.getMethod("charAt" , int.class); //获取String类中的charAt方法,并且指定的是参数列表为一个int类型,的那个方法。
然后用对象去调用获取得到的方法。使用invoke(Object obj,参数) ; 方法实现。
char ch = methodCharAt.invoke(str,1); 这里指的就是通过获得的charAt方法来获取str这个对象 在1位置上的那个字符,打印结果是 b
补充:
(1)Methods[ ] getMethods(); 获取所有公有的方法,返回到一个数组里面。
然后通过对数组的变量,就可以获得所有的方法了。
(2)getMethod();方法 如果是空参函数,那么参数列表就用null