内省技术

最近总感觉自己在学习方面不太专心,所以特意制定了一个学习计划,并通过写博客的方式记录在学习计划过程中自己的所学,今天是学习计划的第一天,希望自己能够给自己的计划开一个好头。

今天的学习内容是内省技术。

一.内省(Introspector)
内省是JDK提供的一套对JavaBean操作的API,是一套基于反射的技术。

  1. 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动作指令中,如果使用即可将表单中的所有数据自动存入bean对象中,其实它的底层就是通过内省技术实现的。

上一篇:JSON —— 序列化与反序列化


下一篇:java Bean封装