Java之旅—反射

Posted gavanwanggw

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java之旅—反射相关的知识,希望对你有一定的参考价值。

  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类中的各个成分反射成对应的类。熟悉以及掌握反射将会帮助我们更好地开发程序。学习研究仍在继续。









以上是关于Java之旅—反射的主要内容,如果未能解决你的问题,请参考以下文章

Java之旅—反射

我的Android进阶之旅NDK开发之在C++代码中使用Android Log打印日志,打印出C++的函数耗时以及代码片段耗时详情

Python学习之旅 —— 基础篇反射冒泡排序算法

反射机制

Kotlin学习之旅解决错误:kotlin.NotImplementedError: An operation is not implemented: Not yet implemented(代码片段

Java反射小练之手写BeanUtils的copyProperties(Upgrade)