-------android培训、java培训、期待与您交流! ----------
一 内省概述
(1)javaBean的概述
内省(Introspector)是 Java 语言对 Bean 类属性、事件的一种缺省处理方法。
JDK中提供了对JavaBean进行操作的一些API,这套API就称为内省
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使用
class Person{ private String name; private int age; public void setName(String name){ this.name = name; } public String getName(){ return this.name; } public void setAge(int age){ this.age = age; } public int getAge(){ return this.age; } Person(){} Person(String name,int age){ this.name = name; this.age = age; } }
一个符合JavaBean特点的类可以当作普通类一样进行使用,但把它当JavaBean用肯定需要带来一些额外的好处,我们才会去了解和应用JavaBean!好处如下:
在Java EE开发中,经常要使用到JavaBean。很多环境就要求按JavaBean方式进行操作,别人都这么用和要求这么做,那你就没什么挑选的余地!
JDK中提供了对JavaBean进行操作的一些API,这套API就称为内省。如果要你自己去通过getX方法来访问私有的x,怎么做,有一定难度吧?用内省这套api操作JavaBean比用普通类的方式更方便。
(2)对javaBean的简单内省操作
1)用eclipse自动生成 ReflectPoint类的setter和getter方法
package cn.itcast.day1;
import java.util.Date; public class ReflectPoint { private Date birthday = new Date(); private int x; private int y; public ReflectPoint(int x, int y) { super(); this.x = x; this.y = y; } public int getX() { return x; } public void setX(int x) { this.x = x; } public int getY() { return y; } public void setY(int y) { this.y = y; } }
2)需求:用内省的方式来读取 ReflectPoint这个javaBean的X属性
直接new一个PropertyDescriptor对象的方式来让大家了解JavaBean API的价值,先用一段代码读取JavaBean的属性,然后再用一段代码设置JavaBean的属性
演示用eclipse将读取属性和设置属性的流水帐代码分别抽取成方法:
只要调用这个方法,并给这个方法传递了一个对象、属性名和设置值,它就能完成属性修改的功能。
package cn.itcast.day1; import java.beans.IntrospectionException; import java.beans.Introspector; import java.beans.PropertyDescriptor; import java.lang.reflect.InvocationTargetException; import org.apache.commons.beanutils.BeanUtils; import org.apache.commons.beanutils.PropertyUtils; 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 value = 7; setProperties(pt1, propertyName, value); System.out.println(pt1.getX()); } private static void setProperties(Object pt1, String propertyName, Object value) throws IntrospectionException, IllegalAccessException, InvocationTargetException { PropertyDescriptor pd2 = new PropertyDescriptor(propertyName,pt1.getClass()); Method methodSetX = pd2.getWriteMethod(); methodSetX.invoke(pt1,value); } private static Object getProperty(Object pt1, String propertyName) throws IntrospectionException, IllegalAccessException, InvocationTargetException { 用内省的方式读取javaBean的X属性 PropertyDescriptor pd = new PropertyDescriptor(propertyName,pt1.getClass()); Method methodGetX = pd.getReadMethod(); Object retVal = methodGetX.invoke(pt1);//调用get方法不接收参数,但是有返回值 return retVal; } }
(3)对javaBean的复杂内省操作
将上述的例子的getProperty方法中换一种用内省的方式读取javaBean的X属性
private static Object getProperty(Object pt1, String propertyName) throws IntrospectionException, IllegalAccessException, InvocationTargetException { /* 用内省的方式读取javaBean的X属性 PropertyDescriptor pd = new PropertyDescriptor(propertyName,pt1.getClass()); Method methodGetX = pd.getReadMethod(); Object retVal = methodGetX.invoke(pt1);//调用get方法不接收参数,但是有返回值 */ /*得到BeanInfo最好采用“obj.getClass()”方式,而不要采用“类名.class”方式,这样程序更通用。 采用遍历BeanInfo的所有属性方式来查找和设置某个RefectPoint对象的x属性。 在程序中把一个类当作JavaBean来看,就是调用IntroSpector.getBeanInfo方法, 得到的BeanInfo对象封装了把这个类当作JavaBean看的结果信息。*/ BeanInfo beanInfo = Introspector.getBeanInfo(pt1.getClass()); PropertyDescriptor[] pds = beanInfo.getPropertyDescriptors(); Object retVal = null; for(PropertyDescriptor pd : pds){ if(pd.getName().equals(propertyName)) { Method methodGetX = pd.getReadMethod(); retVal = methodGetX.invoke(pt1); break; } } return retVal; }
(4)javaBean练习
package com.itheima; /** * 第一题: 写一个方法,此方法可将obj对象中名为propertyName的属性的值设置为value. * public void setProperty(Object obj, String propertyName, Object value){ } * @author wangzhi */ import java.beans.PropertyDescriptor; import java.lang.reflect.Method; public class Test1 { public static void setProperty(Object obj,String propertyName,Object value) throws Exception { PropertyDescriptor pd = new PropertyDescriptor(propertyName,obj.getClass()); Method me = pd.getWriteMethod(); me.invoke(obj, value); } public static void main(String[] args) throws Exception { Person p = new Person("zhuhui",26); System.out.println(p.getName()+":"+p.getAge()); //zhuhui:26 String propertyName1 = "name"; PropertyDescriptor pd1 = new PropertyDescriptor(propertyName1, p.getClass()); //定义一个属性描述符对象 Method method3 = pd1.getReadMethod(); //得到对象pd1的读方法,即为get属性方法。 System.out.println(method3.invoke(p)); //zhuhui Method method = pd1.getWriteMethod(); //得到对象pd1的写方法,即为set属相方法。 method.invoke(p, "justinhui86"); System.out.println(p.getName()+":"+p.getAge()); //justinhui86:26 String propertyName2 = "age"; PropertyDescriptor pd2 = new PropertyDescriptor(propertyName2, p.getClass()); Method method2 = pd2.getWriteMethod(); method2.invoke(p,27); System.out.println(p.getName()+":"+p.getAge()); //justinhui86:27 setProperty(p,"name","justinhui"); System.out.println(p.getName()+":"+p.getAge()); //justinhui:27 } } //定义一个Person类 class Person{ private String name; private int age; public void setName(String name){ this.name = name; } public String getName(){ return this.name; } public void setAge(int age){ this.age = age; } public int getAge(){ return this.age; } Person(){} Person(String name,int age){ this.name = name; this.age = age; } }
二
工具包Beanutils
使用Beanutils工具包操作javaBean
package cn.itcast.day1;
import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Map;
import org.apache.commons.beanutils.BeanUtils;
import org.apache.commons.beanutils.PropertyUtils;
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 value = 7;
setProperties(pt1, propertyName, value);
//在内省例子的基础上,用BeanUtils类先get原来设置好的属性,再将其set为一个新值。
//get属性时返回的结果为字符串,set属性时可以接受任意类型的对象,通常使用字符串。
System.out.println(BeanUtils.getProperty(pt1, "x").getClass().getName());
BeanUtils.setProperty(pt1, "x", "9");
System.out.println(pt1.getX());
/*
//java7的新特性
Map map = {name:"zxx",age:18};
BeanUtils.setProperty(map, "name", "lhm");
*/
//在内省例子的基础上,用BeanUtils类先get原来设置好的属性,再将其set为一个新值。
//get属性时返回的结果为字符串,set属性时可以接受任意类型的对象,通常使用字符串。
BeanUtils.setProperty(pt1, "birthday.time", "111");
System.out.println(BeanUtils.getProperty(pt1, "birthday.time"));
//用PropertyUtils类先get原来设置好的属性,再将其set为一个新值。
//et属性时返回的结果为该属性本来的类型,set属性时只接受该属性本来的类型。
PropertyUtils.setProperty(pt1, "x", 9);
System.out.println(PropertyUtils.getProperty(pt1, "x").getClass().getName());
}
private static void setProperties(Object pt1, String propertyName,
Object value) throws IntrospectionException,
IllegalAccessException, InvocationTargetException {
PropertyDescriptor pd2 = new PropertyDescriptor(propertyName,pt1.getClass());
Method methodSetX = pd2.getWriteMethod();
methodSetX.invoke(pt1,value);
}
private static Object getProperty(Object pt1, String propertyName)
throws IntrospectionException, IllegalAccessException,
InvocationTargetException {
/*
用内省的方式读取javaBean的X属性
PropertyDescriptor pd = new PropertyDescriptor(propertyName,pt1.getClass());
Method methodGetX = pd.getReadMethod();
Object retVal = methodGetX.invoke(pt1);//调用get方法不接收参数,但是有返回值
*/
/*得到BeanInfo最好采用“obj.getClass()”方式,而不要采用“类名.class”方式,这样程序更通用。
采用遍历BeanInfo的所有属性方式来查找和设置某个RefectPoint对象的x属性。
在程序中把一个类当作JavaBean来看,就是调用IntroSpector.getBeanInfo方法,
得到的BeanInfo对象封装了把这个类当作JavaBean看的结果信息。*/
BeanInfo beanInfo = Introspector.getBeanInfo(pt1.getClass());
PropertyDescriptor[] pds = beanInfo.getPropertyDescriptors();
Object retVal = null;
for(PropertyDescriptor pd : pds){
if(pd.getName().equals(propertyName))
{
Method methodGetX = pd.getReadMethod();
retVal = methodGetX.invoke(pt1);
break;
}
}
return retVal;
}
}
三 总结
Web 开发框架 Struts 中的 FormBean 就是通过内省机制来将表单中的数据映射到类的属性上,因此要求 FormBean 的每个属性要有 getter/setter 方法。但也并不总是这样,什么意思呢?就是说对一个 Bean 类来讲,我可以没有属性,但是只要有 getter/setter 方法中的其中一个,那么 Java 的内省机制就会认为存在一个属性,比如类中有方法 setMobile ,那么就认为存在一个 mobile 的属性。
将 Java 的反射以及内省应用到程序设计中去可以大大的提供程序的智能化和可扩展性。有很多项目都是采取这两种技术来实现其核心功能,例如我们前面提到的 Struts ,还有用于处理 XML 文件的 Digester 项目,其实应该说几乎所有的项目都或多或少的采用这两种技术。在实际应用过程中二者要相互结合方能发挥真正的智能化以及高度可扩展性。
------- android培训 、 java培训 、期待与您交流! ----------