内省:

内省的英文单词: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()的属性名—>idsetCPU的属性名是什么?àCPUgetUPS的属性名是什么?à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类型