内省:
内省的英文单词:Introspector(内省,反省)。内省是用来操作JavaBean的一套API,想要理解内省先要知道JavaBean。
JavaBean:
JavaBean是一种特殊的Java类,主要用于传递数据信息,这种java类中的方法主要用于访问私有的字段,且方法名符合某种命名规则。
如果要在两个模块之间传递多个信息,可以将这些信息封装到一个JavaBean中,这种JavaBean的实例对象通常称之为值对象(Value Object,简称VO)。这些信息在类中用私有字段来存储,如果读取或设置这些字段的值,则需要通过一些相应的方法来访问,即set和get方法。
JavaBean的属性是根据其中的setter和getter方法来确定的,而不是根据其中的成员变量。
一个类被当作javaBean使用时,JavaBean的属性是根据方法名推断出来的,它根本看不到java类内部的成员变量。
JavaBean获取属性名的规则:去掉set或get前缀,剩余部分就是属性名,如果剩余部分的第二个字母是小写的,则把剩余部分的首字母改成小的。
例如:
setId()的属性名—>id
setCPU的属性名是什么?àCPU
getUPS的属性名是什么?àUPS
一个符合JavaBean特点的类可以当作普通类一样进行使用,但把它当JavaBean用肯定需要带来一些额外的好处,我们才会去了解和应用JavaBean!好处如下:
1、在Java EE开发中,经常要使用到JavaBean。很多环境就要求按JavaBean方式进行操作,别人都这么用和要求这么做,那你就没什么挑选的余地!
2、JDK中提供了对JavaBean进行操作的一些API,这套API就称为内省。如果要你自己去通过getX方法来访问私有的x,怎么做,有一定难度吧?用内省这套api操作JavaBean比用普通类的方式更方便。
内省的操作方式(3种):
1,使用Introspector.getBeanInfo方法返回BeanInfo类进行内省。(较为复杂)
public class IntroSpectorTest {
public static void main(String[] args) throws Exception{
ReflectPoint pt1 = new ReflectPoint(3, 4);
String propertyName = "x";
System.out.println(getProperty(pt1, propertyName));
setProperty(pt1, propertyName);
System.out.println(pt1.getX());
}
//设置属性值的方法。
private static void setProperty(ReflectPoint pt1, String propertyName)
throws IntrospectionException, IllegalAccessException,
InvocationTargetException {
//对pt1进行内省,了解其所有属性、公开的方法和事件。
BeanInfo bi = Introspector.getBeanInfo(pt1.getClass());
//获取pt1所有属性描述。
PropertyDescriptor[] pds = bi.getPropertyDescriptors();
//遍历所有属性描述。
for(PropertyDescriptor pd : pds){
//获取属性“x”的属性描述。
if(pd.getName().equals(propertyName)){
Method methodSetX = pd.getWriteMethod();
methodSetX.invoke(pt1, 8);
break;
}
}
}
//获取属性值的方法。
private static Object getProperty(ReflectPoint pt1, String propertyName)
throws IntrospectionException, IllegalAccessException,
InvocationTargetException {
//对pt1进行内省,了解其所有属性、公开的方法和事件。
BeanInfo bi = Introspector.getBeanInfo(pt1.getClass());
//获取pt1所有属性描述。
PropertyDescriptor[] pds = bi.getPropertyDescriptors();
Object retVal = null;
//遍历所有属性描述。
for(PropertyDescriptor pd : pds){
//获取属性“x”的属性描述。
if(pd.getName().equals(propertyName)){
//属性x的get方法。
Method methodGetX = pd.getReadMethod();
//调用get方法。
retVal = methodGetX.invoke(pt1);
break;
}
}
return retVal;
}
}
//JavaBean
class ReflectPoint {
private int x;
private int y;
public ReflectPoint(int x, int y) {
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,直接使用PropertyDescriptor类进行内省。
//getProperty方法则写成:
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);
//setProperty方法则写成:
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 = methodGetX.invoke(pt1);
3,使用BeanUtils工具包进行内省操作。
导入外界工具包的过程:
(1,解压--> readme.txt -->A PI --> 找到正确jar包。
(2,eclipse导包:右键工程 --> Build Path --> Configure Build Path --> Add Exteral JARs
lib文件夹 --> lib下粘贴jar包 --.> 右键jar包 -->
Build Path --> Add to Build Path
注意:BeanUtils工具包用到了阿帕奇的日志包,所以导入日志包后工具才能正常使用。
代码范例:
System.out.println(BeanUtils.getProperty(pt1,”x”)); //返回值string类型
BeanUtils.setProperty(pt1,”x”,”9”); //自动将“9”转换类型并set进去。
BeanUtils.setProperty(pt1,”birthday.time”,”9”); //支持属性的级联。Birthday属性定义时初值不能为空,否则空指针异常。
BeanUtils还可以操作Map集合:
(并且BeanUtils提供了Map与JavaBean相互转换的方法)
Map map = {name:”cxy”,age:18};//jdk1.7新特性。
BeanUtils.setProperty(map,”age”,”20”);
PropertyUtils类与BeanUtils类的区别:
PropertyUtils的内省操作遵循属性原有的类型,不进行类型转换。
PropertyUtils.setProperty(pt1,”x”,9);
BeanUtils.getProperty(pt1,”x”); //返回值是Integer类型