内省->了解JavaBean
IntroSpector-->JavaBean-->特殊的Java类
特殊的规则是:
class Person
{
private int age;
public int getAge(){
return this.age;
}
public void setAge(int age){
this.age = age;
}
}
JavaBean的属性是根据getter、setter方法名来的,
Age-->如果第二个字母是小的,则第一个字母变成小写的-->age
gettime-->time
setTime-->time
getCPU-->CPU
·JavaBean是一种特殊的Java类,主要用于传递数据信息,这种java类中的方法主要用于访问私有的字段,且方法名称符合某种命名规则。
·如果要在两个模块之间传递多个信息,可以将这些信息封装到一个JavaBean中,这种JavaBean的实例对象通常称之为值对象(Value Object,简称VO)。这些信息在类中用私有字段来存储,如果读取或设置这些字段的值,则需要通过一些相应的方法来访问,大家觉得这些方法的名称叫什么好呢?JavaBean的属性是根据其中的setter和getter方法来确定的,而不是根据其中的成员变量。如果方法名为setId,中文意思即为设置id,至于你把它存在哪个变量上,用管吗?如果方法名为getId,中文意思即为获取id,至于你从哪个变量上取,用管吗?去掉set前缀,剩余部分就是属性名,如果剩余部分的第二个字母是小写的,则把剩余部分的首字母改成小的。
-setId()的属性名->id
-isLast()的属性名->last
-setCPU的属性名是什么?->CPU
-getUPS的属性名是什么?->UPS
总之,一个类被当做JavaBean使用时,JavaBean的属性是根据方法名推断出来的,它根本看不到java类内部的成员变量。
·一个符合JavaBean特点的类可以当做普通类一样进行使用,但把它当JavaBean用肯定需要带来一些额外的好处,我们才会去了解和应用JavaBean!好处如下:
-在Java EE开发中,经常要使用到JavaBean。很多环境就要求按JavaBean方式进行操作,别人都这么用和要求这么做,那你就没什么挑选 的余地!
-JDK中提供了对JavaBean进行操作的一些API,这套API就称为内省。如果要你自己去通过getX方法来访问私有的x,怎么做,有一定难度吧?用内省这套api操作JavaBean比用普通类的方式更方便。
操作技巧:重构抽取方法eclipse快捷键alt+shift+M
内省综合案例
·演示用eclipse自动生成ReflectPoint类的setter和getter方法。快捷键:alt+shift+s +r
·直接new一个PropertyDescriptor对象的方式来让大家了解JavaBean API的价值,先用一段代码读取JavaBean的属性,然后再用一段代码设置JavaBean的属性。
·演示用Eclipse将读取属性和设置属性的流水帐代码分别抽取成方法:快捷键:alt+shift+M
-只要调用这个方法,并给这个方法传递类一个对象、属性名、和设置值,它就能完成属性修改的功能。
-得到BeanInfo最好采用“obj.getClass()”方式,而不要采用“类名.class”方式,这样程序更通用。
·采用遍历BeanInfo的所有属性方式来查找和设置某个ReflectPoint对象的x属性。在程序中把一个类当做JavaBean来看,就是调用IntroSpector.getBeanInfo方法,得到的BeanInfo对象封装类把这个类当做JavaBean看的结果信息。
对JavaBean的简单内省操作
package lqq.heima;
import java.beans.IntrospectionException;
import java.beans.PropertyDescriptor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class IntroSpectorTest {
/**
* @param args
*/
public static void main(String[] args) throws Exception {
// TODO Auto-generated method stub
ReflectPoint pt1 = new ReflectPoint(3,5);
String propertyName = "x";
//"x"-->"X"-->"getX"-->MethodGetX-->
Object retVal = getProperty(pt1, propertyName);
System.out.println(retVal);
Object setValue = 7;
setProperty(pt1, propertyName, setValue);
System.out.println(getProperty(pt1, propertyName));
}
private static Object getProperty(Object pt1, String propertyName)
throws IntrospectionException, IllegalAccessException,
InvocationTargetException {
PropertyDescriptor pd = new PropertyDescriptor(propertyName, pt1.getClass());
//Method methodGetX = pd.getReadMethod();
Object retVal = pd.getReadMethod().invoke(pt1);
return retVal;
}
private static void setProperty(Object pt1, String propertyName,
Object setValue) throws IntrospectionException,
IllegalAccessException, InvocationTargetException {
PropertyDescriptor pd2 = new PropertyDescriptor(propertyName, pt1.getClass());
Method methodSetX = pd2.getWriteMethod();
methodSetX.invoke(pt1, setValue);
}
}
以上方法中用到的PropertyDescriptor类的构造方法要注意其特性:
PropertyDescriptor
public PropertyDescriptor(String propertyName,
Class<?> beanClass)
throws IntrospectionException通过调用 getFoo 和 setFoo 存取方法,为符合标准 Java 约定的属性构造一个 PropertyDescriptor。因此如果参数名为 "fred",则假定 writer 方法为 "setFred",reader 方法为 "getFred"(对于 boolean 属性则为 "isFred")。注意,属性名应该以小写字母开头,而方法名称中的首写字母将是大写的。
参数:
propertyName - 属性的编程名称。
beanClass - 目标 bean 的 Class 对象。例如 sun.beans.OurButton.class。
抛出:
IntrospectionException - 如果在内省期间发生异常。
对JavaBean的复杂内省操作:主要用到Introspector类的静态方法getBeanInfo()得到一个BeanInfo对象,然后调用BeanInfo的getPropertyDescriptors方法,或者这个JavaBean中的所有的属性描述PropertyDescriptor[],然后遍历些属性描述通过比较pd.getName()来找出自己所需要的属性,然后再操作这个属性对应的方法。代码如下:
private static Object getProperty(Object pt1, String propertyName)
throws IntrospectionException, IllegalAccessException,
InvocationTargetException {
/*
PropertyDescriptor pd = new PropertyDescriptor(propertyName, pt1.getClass());
//Method methodGetX = pd.getReadMethod();
Object retVal = pd.getReadMethod().invoke(pt1);
*/
Object retVal = null;
BeanInfo beaninfo = Introspector.getBeanInfo(pt1.getClass());
PropertyDescriptor [] pds = beaninfo.getPropertyDescriptors();
for(PropertyDescriptor pd:pds){
if(pd.getName().equals(propertyName))
{
retVal = pd.getReadMethod().invoke(pt1);
break;
}
}
return retVal;
}
使用BeanUtils工具包操作JavaBean
BeanUtils工具包
·演示用eclipse如何加入jar包,先只是引入beanutils包,等程序运行出错后再引入logging包。
·在前面的内省例子的基础上,用BeanUtils类先get原来设置好的属性,再将其set为一个新值。
-get属性时返回的结果为字符串,set属性时可以接受任意类型的对象,通常使用字符串。
·用PropertyUtils类先get原来设置好的属性,再将其set为一个新值。
-get属性时返回的结果为该属性本来的类型,set属性时只接受该属性本来的类型。
关键代码如下:
//BeanUtils && PropertyUtils
//BeanUtils 没有限制设置传入参数的类型要与Bean中的类型相同,设置与返回都是以String类型来处理的。
BeanUtils.setProperty(pt1, propertyName, 9);
System.out.println("BeanUtils.getProperty: "+BeanUtils.getProperty(pt1, propertyName));
BeanUtils.setProperty(pt1, "birthday.time", 111);
System.out.println("birthday.time:"+BeanUtils.getProperty(pt1, "birthday.time"));
System.out.println("birthday:"+BeanUtils.getProperty(pt1, "birthday"));
//PropertyUtils 限制了设置传入参数的类型要与Bean中的类型相同,设置与返回都是以Bean中的实际类型来处理的。
PropertyUtils.setProperty(pt1, propertyName, 9);
System.out.println(PropertyUtils.getProperty(pt1, propertyName).getClass().getName());
/*
java7的新特性
Map map = {name:"zxx",age:18};
BeanUtils.setProperty(map,"name","lhm");
*/ |