java零基础入门-高级特性篇(十五) 类加载与反射 2
继续上一章的内容,当我们获取到了类的Class对象,下面就可以用Class对象来干大事了。
反射
反射是什么?反射是一种能力,能够在程序运行的过程中获取类的信息,创建对象,使用方法的能力。
看名字,首先想到的是光的反射。我们能在镜子里面看见自己,因为光线照射在镜子上,然后反射到我们眼睛里,所以我们就能在镜子里看见自己。这样,我们可以通过观察镜子来获取到我们自身的信息。java中的反射也有类似的特性,当一个java类加载到jvm后,生成了Class对象,这时候Class对象就像镜子一样,当我们在java应用程序中需要获取到已加载的java类的信息时,通过反射这个功能,就可以做到。
反射获取信息
反射包提供了很多的类型来获取类的信息,上例中使用了最常用的几种类型,分别是Constructor构造器类型,Method方法类型,Field成员变量类型,除了这几种常用类型,还有各种方法用于判断枚举类,注解类,接口等等,同学们可以自行查阅API。上例中最重要的就是第一行获取Class对象的方法,Class.forName()。一旦获取到了Class对象,就可以通过该对象“反射”出一个类的各种信息。
反射创建对象
在写代码的过程中,最常用的就是通过new关键字创建对象。但是反射给我们提供了另一种创建对象的思路,即是在运行时动态的创建对象。通过Class对象的newInstance()方法即可创建对象。上例中就是通过newInstance()方法创建对象的具体过程,需要注意的是,如果不指定构造器的话,newInstance()方法只能根据无参构造器创建对象。如果需要使用有参构造器,必须首先通过Class对象获取指定构造器对象,再通过获取的构造器来创建对象。
在获取有参构造器的时候,也要注意一点。由于构造器是可以重载的,所以在获取构造器的时候必须指定构造器的参数类型,这样才能准确的找到我们需要的构造器。这也是重载构造器时,必须要指定不同类型或者不同数量的参数才能重载构造器的原因,在使用构造器的时候,必须有办法能找出“唯一”符合要求的那个构造器。
反射调用方法
反射调用方法的流程也与上面差不多,首先是获取Class对象,有了Class对象再创建实例对象,然后获取类的方法,最后用反射获取的方法类调用方法。获取方法和获取构造器有点区别,构造器的名字都是与类名一致的,所以无需传递构造器的名称,但是方法的名称是自定义的,所以在通过Class对象获取方法的时候,不但需要传入方法参数的类型,还需要传入方法名。
在调用方法的时候,和普通对象调用方法时一样,都需要一个已经被创建好的对象。区别是通过new关键字创建的对象通过“.”来调用方法,而通过反射来调用方法是需要将实例对象作为参数,传递给Method类的对象的invoke()方法。这样就能通过反射调用方法了。
反射使用举例
讲了这么多反射的使用方法,具体该怎么用呢?下面用一个例子来说一下。
在做项目的过程中,特别是web项目,有一个绕不开的问题就是不同类型直接进行值得传递。比如在页面上输入的数据格式与实际数据库需要储存的数据格式不同,这时候就需要把数据在不同的对象之间进行传递。
为什么是不同的对象?因为后台服务必须为前台传递过来的数据准备好一个对象,前台传过来什么类型,后台就需要用这个格式来接收。比如页面上的生日是个字符串格式“2011-11-11”,后台就需要一个特定的对象,这个对象需要用字符串类型的birthday变量来接收它,这个特定的对象通常叫做VO-视图对象。
但是在数据库里面储存的时候,有时候直接储存字符串的日期格式会有不便,比如有一个业务需要按照年龄排序,如果是字符串格式的日期格式就会比较麻烦,需要很多额外的处理。所以有时候我们需要存的是Long类型,用来做比较的时候会十分方便,这就需要将页面传递的字符串转换为Long类型,用另一个birthday为Long类型的对象来接管。
然后程序员们就大笔一挥,不就是个类型转换么,很简单啊,看我的,我分分钟搞定~
这个做法是非常常见的,工作中也会被大量使用,但是不建议这么做,因为这个类的成员变量还算少,类型转换也不算复杂,要是属性多,从头一个个转下来,会有一大片一大片的转换代码,体力劳动必须想办法避免。怎么办?体力劳动封装成方法,传入一个vo对象,一个dto对象,在方法里进行自动转换就完了。这个方法里必须能够获取到两个对象的所有信息才能转换,这时候反射就派上用场了。
这个转换的思路就是,首先从vo中遍历出所有的属性,然后从属性中获取对应的值,再根据属性名为dto中对应的属性赋值。要注意VO和DTO中的属性都是私有的,所以在获取值和设置值之前都必须通过setAccessible()将该属性设置为可访问,这样就可以直接访问私有变量,而无需通过方法中的set和get来获取和设置属性值。但是这样做要注意,因为改变访问权会破坏程序的封装性。
这里只是为了讲解反射的作用简单的写了一个方法,其中还有不少的缺陷。其实在工作中有更好的方法完成这个任务,比如Apache提供了封装好的JavaBean操作工具--BeanUtils。这个工具功能更加强大,并且提供了大量十分方便高效的工具方法,比如BeanUtils.copyproperties()这个方法就是上例的最好解决方法,并且在转换中有更好的效率。