最近总感觉自己在学习方面不太专心,所以特意制定了一个学习计划,并通过写博客的方式记录在学习计划过程中自己的所学,今天是学习计划的第一天,希望自己能够给自己的计划开一个好头。
今天的学习内容是内省技术。
一.内省(Introspector)
内省是JDK提供的一套对JavaBean操作的API,是一套基于反射的技术。
- JavaBean属性计算方法
通过Introspector获得JavaBean类相关信息
通过BeanInfo获得方法描述器(getMethodDescriptors())和属性描述器(getPropertyDescriptors())
通过getPropertyDescriptors()我们可以得到一个Bean类的属性
所以,首先我们创建一个Peron类
public class Person {
private String name;
private String city;
private String hobby;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getCity() {
return city;
}
public void setCity(String city) {
this.city = city;
}
public String getHobby() {
return hobby;
}
public void setHobby(String hobby) {
this.hobby = hobby;
}
}
然后得到该类的属性描述器,并打印数组长度,会发现数组长度为4,而Person类中只有3个属性
@Test
public void demo1() {
// 通过Introspector获得BeanInfo信息
try {
BeanInfo beanInfo = Introspector.getBeanInfo(Person.class);
// 获得所有JavaBean的属性描述器
//每一个属性描述器代表JavaBean的一个属性
PropertyDescriptor[] propertyDescriptors = beanInfo
.getPropertyDescriptors();
//计算JavaBean的属性
System.out.println(propertyDescriptors.length);
} catch (IntrospectionException e) {
e.printStackTrace();
}
}
所以,这里要注意的是,JavaBean的属性不是通过成员变量计算的,而是通过get和set方法计算的
例如:
你有一个方法getName(),它就会去掉get将Name首字母小写得到一个name属性
你有一个方法getAge(),它就会去掉get将Age首字母小写得到一个age属性
以此计算。
此时又会有疑问了,那按照这样的计算方法,get和set方法只有三个,计算出的属性也应该只有三个啊,我们可以遍历属性描述器并打印名字
@Test
public void demo1() {
// 通过Introspector获得BeanInfo信息
try {
BeanInfo beanInfo = Introspector.getBeanInfo(Person.class);
// 获得所有JavaBean的属性描述器
//每一个属性描述器代表JavaBean的一个属性
PropertyDescriptor[] propertyDescriptors = beanInfo
.getPropertyDescriptors();
//计算JavaBean的属性
System.out.println(propertyDescriptors.length);
//遍历
for(PropertyDescriptor prDescriptor : propertyDescriptors){
System.out.println(prDescriptor.getName());
}
} catch (IntrospectionException e) {
e.printStackTrace();
}
}
得到的结果:
会发现控制台打印了第四个属性class,而在Person类中并没有被找到。原来,每个对象都继承自Object类,从而继承了Object对象的getClass方法,根据JavaBean属性的计算方法,从而得到第四个属性class,问题由此就迎刃而解了。
那内省到底有什么作用呢?它的意义何在呢?通过一个案例来演示(将Map的值保存到相应对象的属性中):
@Test
public void demo2() {
// 将Map的值保存到相应的对象属性中
Map<String, String> map = new HashMap<String, String>();
map.put("name", "mary");
map.put("city", "北京");
map.put("hobby", "music");
Person person = new Person();
//保存数据
saveMapValueToObject(map, person);
System.out.println(person.getName());
System.out.println(person.getCity());
System.out.println(person.getHobby());
}
// 将Map中的key与Object中的属性进行匹配,将Map对应的value保存至Object的属性中
private void saveMapValueToObject(Map<String, String> map, Object obj) {
try {
// 内省获得Javab的信息
BeanInfo beanInfo = Introspector.getBeanInfo(obj.getClass());
// 通过beanInfo获得所有属性
PropertyDescriptor[] propertyDescriptors = beanInfo
.getPropertyDescriptors();
// 遍历每一个属性
for (PropertyDescriptor pDescriptor : propertyDescriptors) {
// 用属性去Map中寻找对应的key
String name = pDescriptor.getName();
if (map.containsKey(name)) {// 属性在Map中存在对应的key
String value = map.get(name);
// 通过属性描述器,获得写入该属性的方法
Method setMethod = pDescriptor.getWriteMethod();
// 利用反射设置value
setMethod.invoke(obj, value);
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
通过控制台信息可以知道,Map集合中的数据成功地被保存到了Person类对应的属性中,该案例即是内省技术的应用。
需要知道的是,在JSP动作指令jsp:useBean</jsp:useBean>中,如果使用<jsp:setProperty property = "*"/>即可将表单中的所有数据自动存入bean对象中,其实它的底层就是通过内省技术实现的。