1. 概念
反射就是将java类中的各种成分映射成对应的java类。之前我们已经讲过了Class类,也明确了一个java类中用一个Class类的对象来表示,一个类中的组成部分有:成员变量,方法。构造方法,包等等信息也用一个个的java类来表示。
举个生活中的样例,生活中我们常常见到汽车,假设说汽车是一个类的话。那么汽车中的发动机、变速箱等等也是一个个的类。表示java类的Class类显然要提供一系列的方法,来获得当中的变量。方法,构造方法,修饰符,包等信息,这些信息就是用对应类的实例对象来表示的,就是Field,Method,Contructor以及Package等等。
2. 机制及应用场合
在执行状态中,对于不论什么一个类,都可以知道这个类的所以的属性和方法;对于不论什么一个对象,都可以调用它的随意一个方法和属性。这样的动态获取信息以及动态调用对象的方法就是反射,java反射的动态机制是Reflection,指的就是可以用于执行期间的载入和探知。使用编译期间全然未知的classes。
换言之。java程序课可以载入一个执行时才知道的class,获悉其完整构造(不包括method定义)。
在编译时根本无法知道该对象或者类可能属于哪些类。程序仅仅是依靠执行时的信息来发现该对象和类的真实的信息。
3. 作用
通过反射我们能够使程序代码訪问装载到JVM中的类的内部信息,获取已装载类的属性信息、方法、构造方法的信息。
我们前面的介绍到了Class类是java反射机制的起源和入口,用户获取与类相关的各种信息,提供了获取类信息的相关的方法,Class类继承自Object类,Class类是全部类的共同的图纸。
Class类的一些方法例如以下图所看到的:
4. 各种反射的应用
Ø 构造方法 Constructor类
一个类有多个构造方法,依据參数的个数和类型不用来差别,java中的构造方法是没有顺序的;利用构造方法类型的方法能够反射得到对应的类;由构造方法能够得到对应的实例对象,依据方法的反射得到这个构造方法究竟是哪个类的构造方法。如以下所看到的:
<span style="font-family:Microsoft YaHei;font-size:14px;"><strong>Constructor constructor1 = String.class .getConstructor(StringBuffer.class); String str2 = (String) constructor1 .newInstance(new StringBuffer("abc"));</strong></span>
仅仅是知道是个构造方法,可是不知道是哪个构造方法,仅仅有在执行的时候才干知道
上例中我们看到存在两个StringBuffer,
第一个表示用的是String众多中哪个构造方法,由上例可知。用的是參数为StringBuffer的构造方法。
表示的是获得方法时要用到类型。
第二个表示是用的时候必须传入一个stringbuffer的对象;利用构造方法的newInstance反射能够得到这个类的实例对象。创建这个类型实例对象,从而将构造方法的同样的类型參数进行传入表示调用了某个方法。调用获取的方法时要用到上面同样类型的实例对象。
值得注意的是:參数必须是匹配的。假设不匹配,则会报错。编译的时候仅仅是运行语法检查,并不运行等号右边的代码。仅仅是在运行的时候才干知道其是一个String,故而以下的要加上(String)。
Ps:除了构造方法具有newInstance()方法外,Class相同具有,也就是Class.newInstance(),比方:<span style="font-family:Microsoft YaHei;font-size:14px;"><strong>Stringobj=(String)Class.forName(“java.lang.String”).newInstance();</strong></span>
该方法内部先得到默认的构造方法,然后用该构造方法创建实例对象。
Ø 成员变量 Filed类
定义一个ReflectPoint类,然后在一个类中进行调用。
<span style="font-family:Microsoft YaHei;font-size:14px;"><strong>ReflectPoint pt1 = new ReflectPoint(3, 5); Field fieldY = pt1.getClass().getField("y"); System.out.println(fieldY.get(pt1)); System.out.println(fieldY); </strong></span>
Field fieldY = pt1.getClass().getField("y");假设你觉得这个表达式得到的值是5,那就错了,fieldY不是对象身上的变量。而是类上的,要用它去取某个对象上相应的值。
fieldY.get(pt1),上面我们知道fieldY是某个类上的变量,返回类型是个类型,也能够说是一个类。利用它的get方法能够反射得到相应的实例对象而且获取实例对象上相应的值。
Ø 成员方法的反射
就是一个类中定义的方法。Method类代表某个类中的一个成员方法。将一个类中各个成分反射成其相应的java类就是反射。
得到类中的一个方法,利用getMethod得到的是一个Method类型的类。而且能够在括号里匹配对应的參数类型与类名。反射方式得到字节码中的方法,再拿出这种方法去作用于某个对象。
<span style="font-family:Microsoft YaHei;font-size:14px;"><strong>Method methodCharAt = String.class.getMethod("charAt", int.class); System.out.println(methodCharAt.invoke(str1, 1)); </strong></span>
方法的调用肯定是基于某个对象的:
1.也就是在对象str1身上调用了定义charAt方法(methodCharAt)方法,后面的1是方法的參数
2.利用反射的方式得到字节码中的方法,再拿出这种方法作用于某个对象。Invoke表示的是调用。
3.也就是方法运行了调用的动作。invoke是这种方法对象身上的方法。
故而invoke中第一个即为对应的实例对象,第二个为參数,无论參数有无,均称之为參数。
举个简单的样例:
画圆或者火车停车仅仅是人发信号,操作由其本身来运行。方法是对象本身在调用,而人仅仅是发了个通知或者信号,运行的动作仅仅是自己的动作。
人的作用就是发信号和通知而已,也就是说谁拥有数据谁就是专家。也就是对象本身。
什么方法调用时是不须要对象?答案是静态方法。静态方法调用的时候不须要对象,所以不须要对象的肯定是静态方法。
运用方法的方法反射得到某个实例对象。
<span style="font-family:Microsoft YaHei;font-size:14px;"><strong>Method methodCharAt = String.class.getMethod("charAt", int.class); System.out.println(methodCharAt.invoke(null, 1));</strong></span>
5. 小结
反射就是将java类中的各个成分反射成对应的类。熟悉以及掌握反射将会帮助我们更好地开发程序。学习研究仍在继续。