本文涉及重要概念:反射、内省、BeanUtils工具、JSP标签。
这几个概念看起来相关性不大,其实都是用于控制JavaBean的方法。其中,前面三种都是在java(例如在Servlet)文件中访问和修改JavaBean的方法。而JSP标签则是在jsp文本中访问JavaBean的方法。
在这之前,如果我们想要操作属性,一般情况下我们先是先将类实例化来得到一个对象。然后通过setter和getter方法对这个对象的属性进行操作,
当然,这是访问属性,如果直接用对象来调用方法也没什么不妥。
虽然JSP标签方法看起来繁杂(主要是指标签使用方面),但是它的原理确实最简单的,和上面类似,它对属性的操作本质上还是直接通过对象的getter和setter方法来进行实现(具体的标签的内容将在后面描述)。
而其他三个都是采用不同方式来进行属性的访问。
按照我的理解,反射功能最为强大,它允许通过一个实例化对象找到类的完整信息并得到Class类。而Class类除了能操作属性还能操作方法。内省则是JDK中专门设计用于操作Java对象属性的一套API。BeanUtils是Apache软件基金会提供的一套能动态访问Java对象属性的API,在这三者中最为简洁。这三个类都需要依赖于其他的类来进行对象属性的设置,并且不管是哪一种方法,原有的对象都从属性的调用者变成了参数。而真正的方法的调用者是我们所依赖的这些新的对象。反射中所依赖的是Class类调用getDeclaredField(String<属性名>)方法得到的Field类、内省中所依赖的主要是PropertyDescriptor类,而BanUtils工具中所依赖的主要是BeanUtils类。
但是BeanUtils类不需要实例化就可以调用方法,而反射和内省的这些类都需要实例化。
反射:
任何对象通过getclass()方法它就可以得到一个class类的实例,并且通过这个class类的方法就可以获取类的完整信息。
假如我有一个苹果对象、一个猫对象和一个手机对象,它们各自使用getclass()都可以返回一个class对象。尽管这三个对象本身有不同的属性和方法。但是它们返回的class对象却有相同的方法和属性(不过这些值不同)。通过对象调用getClass()获得一个Class类的实例,需要先创建出一个对象(先要有一个苹果/猫/手机)。
除了使用对象.getClass()方法来获得class实例以外,还可以通过Class类的静态方法forName(),用类的全路径名获取一个Class实例。这种方法不需要提前建立一个(苹果/猫/狗)对象。
Class clazz = Class.forName("cn.itcast.chapter08.reflection.Person");
这里是直接由于Class类来创建Class实例,并在参数中引入具体的类(例如这里的Person类)的路径。
Class类调用forName(<Person类的路径>)方法--->创建关于 Person类的clazz实例-----> clazz实例可以创建Person类的实例。
Person p= (Person)clazz.newInstance();
这段话看起来有一点绕。
使用clazz创建对象时,如果有多个构造方法,那么情况更复杂,需要使用getConstructor()方法得到一个构造方法对象(实际上是Constructor对象)的数组,然后由数组中对应的某个Constructor对象(通过下标选取)来创建对象。(在我看来,这里数组中的每个cons元素都可以理解为选定了特定构造方法的clazz)。
Constructor cons[] = clazz.getConstructors();
Person p = (Person)cons[0].newInstance("李芳",30);
和通过对象直接调用setter或者getter来修改属性不同,现在用了反射。要修改对象的属性,对象无需调用任何的方法,只需要老老实实作为参数进行传递就行了。
具体的属性会用一个Field类型的对象来表示,而具体的方法会用一个Method对象来表示。这些对象都是通过class类来生成。
Field ageField = clazz.geetDeclaredField("age");
ageField.setAccessible(true);
ageField.set(p,20);
反射获取方法部分示例代码如下所示:
Method md = clazz.getMethod("sayHello",String.class,int.class);
String result =(String)md.invoke(clazz.newInstance())
也即是说,不管是通过反射调用属性还是方法,原本是通过该类的实例化对象来调用,而现在全部可以通过class类的实例对象(例如clazz)来完成。并且属性、方法全部都变成了对象(age属性生成一个Field对象,sayHello方法生成一个Method对象),并且这些对象全部都可以通过clazz一步得到,而原来的实例对象从方法和属性的执行者,现在变成了参与者
。
内省
内省是专门用于操作java对象的属性,它比反射技术更加便捷。
反射可以实现操作方法、属性的功能,而这里内省专门操作java对象的属性,从功能上看略差一筹。但是在属性的操作方面,它有它的便捷之处。在内省中也运用到了反射中的class类。
Person p = new Person();
PropertyDescriptor pd = new PropertyDescriptor("name",p.getclass());
Method methodName = pd.getWriterMethod();
methodName.invoke(p,"小明");
同样是访问属性,这里的思路跟用反射访问属性的思路就不同了。上面说过反射是利用class类的一个实例clazz来调用getDeclaredField(String name)方法来得到一个关于某个属性Field对象,这个Filed对象就可以直接更改对应的属性值。
而这里p.getClass()同样生成一个class类的实例,但是呢不直接利用这个实例,而是继续把这个class实例和我们的属性值作为参数创建出一个PropertyDescriptor对象。
这两者的区别,如果是第一种方法,那么当我要访问多个属性,那我还是只需要一个clazz对象。但如果是通过内省的方法,每次访问一个属性,都需要新创建一个Class类实例对象,并作为参数用于生成新的PropertyDescriptor对象。
这还没完,如果想要对属性进行更改。还需要调用getWriteMethod()得到一个method对象,这个method对象才能够对属性进行更改,如果只是进行读取,那么就调用getReadMethod()。说句实在话,我看不出来哪里便捷了。
Beanutils工具
beanutils提供了比反射和内省更为简单的操作,Beanutils工具包还需要一个logging包来配合使用。logging包中包装了各种日志API的实现。BeanUtils工具中最核心的是org.apache.commons.beanutils包下的BeanUtils类。BeanUtils直接调用setProperty方法就可以搞定属性的修改。
Person p = new Person();
BeanUtils.setProperty(p,"name","Jack");
BeanUtils.getProperty(p,"name");
并且能够通过populate方法同时修改多个属性。
Map<String,Object>map = new HashMap<String,Object>();
map.put("name","张三");
map.put("age",10)
BeanUtils.populate(p,map)
但是我们同样也要意识到,这里只是访问属性方便。方法就不见得能访问了。
JSP标签访问JavaBean
jsp文件访问JavaBean和更改其属性可以通过标签来进行实现。但是其对属性的操作本质上还是利用了这个对象直接调用getter和setter的方法来进行实现。因此,接下来的内容是对标签用法的一些介绍。
<jsp:useBean>
表示可以在某个指定的域的范围中查找一个指定名称的JavaBean对象。如果已经存在对象则返回该对象,否则实例化一个JavaBean对象并储存在指定的域中。
<jsp:setProperty>
标签有name、property、param和value四个属性。其中name和property好理解,表明要修改的是名为name的JavaBean实例对象的property属性。这时候,如果我们访问这个.jsp文件,URL链接后面的参数如果和property的名字相对应的,会自动赋值
到这个属性。如果参数名字和property名字不同但是仍然想赋值,那么可以用param进行映射。如果就算参数相同也不希望进行赋值的,可以使用value属性为其赋值,这样就不受URL请求中的参数的影响。
JSP开发模型的Model1与Model2的区别的理解
JSP Model1:JSP+JavaBean
JSP Model2:Servlet + JSP + JavaBean
可以看到,两种模型都使用了JavaBean技术。不同之处在于JspModel1模型没有引入专门的Servlet。因此控制逻辑都在.jsp中进行实现,不可避免地.jsp页面中含有大量的代码片段。
而JspModel2则是将部分代码片段抽离出来,封装到Servlet中,只留下一些与显示相关的代码段。实现了 页面显示、流程控制和业务逻辑
的分离。
说的通俗些,.jsp曾经是一个无所不能的存在,但是我们不忍心看它忙的焦头烂额,所以我们用JavaBean来帮它控制数据库。再使用Servlet帮他处理一些业务逻辑,这样也有利于代码的维护。