JavaBean和Map的相互转换

一、JavaBean

1.什么是JavaBean?

JavaBean其实就是一种遵循特定写法的类,必须遵循一定的规范:

  • 类必须由public修饰,并且保证有公共的无参数的构造器
  • 包含操作属性的的public方法,getter setter方法
  • 字段是私有的

一个JavaBean是由3部分组成,分别是属性、方法、和事件。注意这里的属性是property,表示的是状态,并不是字段(Field)我们在刚开始学面向对象的时候其实有的时候也是用到了这个规范,我们写的getter和setter方法,比如public void setName(String name){ this.name = name;}属性其实就是name,在比如public int getAge(){return 19;},属性其实就是age。一般的,我们写getter个setter方法用的是IDE自己生成的,属性和字段同名。一定要区分属性和字段,它们并不是一个东西,也存在属性名和字段名不同。

2.内省机制

内省机制的作用就是用来查看和操作JavaBean的属性。

  • 获取JavaBean的属性名
  • 通过getter方法给属性获取值,通过setter给属性设置值

常用的API

Introspector类中

static BeanInfo getBeanInfo(Class<?> beanClass)
在 Java Bean 上进行内省,了解其所有属性、公开的方法和事件。
static BeanInfo getBeanInfo(Class<?> beanClass, Class<?> stopClass)
在给定的“断”点之下,在 Java Bean 上进行内省,了解其所有属性和公开的方法。 

可以看到在Introspector类中有两个静态方法可以获得获取JavaBean的信息,一般的第一个会得到父类Object.class中的getClass方法,第二种会屏蔽掉Object类中的方法。返回一个BeanInfo,在BeanInfo接口中有一个方法可以获得bean的描述器对象

 MethodDescriptor[] getMethodDescriptors() 获得 beans MethodDescriptor。
 PropertyDescriptor[] getPropertyDescriptors()  获得 beans PropertyDescriptor。 

分别返回的是各自的描述器对象

 Method getReadMethod()  获得应该用于读取属性值的方法。 就是getter方法
 Method getWriteMethod()  获得应该用于写入属性值的方法。 就是setter方法
public class User {
    private String name;
    private int age;
    private boolean man;
    public String getName() {
        return name;
    }
    ...省略剩余getter setter方法
    @Override
    public String toString() {
        return "User [name=" + name + ", age=" + age + ", man=" + man + "]";
    }
    public static void main(String[] args) throws Exception {
        //获得JavaBean的描述信息
        BeanInfo info = Introspector.getBeanInfo(User.class,Object.class);
        //获取JavaBean的属性描述器对象,返回的是一个描述器数组
        PropertyDescriptor[] pds = info.getPropertyDescriptors();
        //利用反射创建这个类的对象,当然这只是测试一下
        User u = User.class.newInstance();
        //对描述器对象进行迭代
        for (PropertyDescriptor pd : pds) {
            //获取属性名称 和属性的类型
            System.out.println(pd.getName()+","+pd.getPropertyType()());
             //获取getset方法
            Method get = pd.getReadMethod();
            Method set = pd.getWriteMethod();
            System.out.println("getter方法:"+get);
            System.out.println("setter方法:"+set);

            //用反射给name属性设置值
            if("name".equals(pd.getName())){
                Method setter = pd.getWriteMethod();
                setter.invoke(u, "xx");
                Method getter = pd.getReadMethod();
                Object o = getter.invoke(u);//获取名字
                System.out.println(o);
            }
        }
        System.out.println(u);
    }
}

二、JavaBean和Map的相互转化

我们可以发现javabean和map之间有很多相似之处.在map中是一对键值对key-value,而在JavaBean中是属性名-属性值。现在对这两种进行相互转换

1.JavaBean转化为Map

思路:把bean转化为map,不过就是把bean中的属性名给map的key,把属性值给map中的value。在程序上表达,提供一个方法,传入的是一个JavaBean对象,返回的是一个Map。

上代码:

//把JavaBean转化为map
public static Map<String,Object> bean2map(Object bean) throws Exception{
    Map<String,Object> map = new HashMap<>();
    //获取JavaBean的描述器
    BeanInfo b = Introspector.getBeanInfo(bean.getClass(),Object.class);
    //获取属性描述器
    PropertyDescriptor[] pds = b.getPropertyDescriptors();
    //对属性迭代
    for (PropertyDescriptor pd : pds) {
        //属性名称
        String propertyName = pd.getName();
        //属性值,用getter方法获取
        Method m = pd.getReadMethod();
        Object properValue = m.invoke(bean);//用对象执行getter方法获得属性值

        //把属性名-属性值 存到Map中
        map.put(propertyName, properValue);
    }
    return map;
}

2.把Map转化为JavaBean

思路:map转化为javabean,不就是把map的value设置给属性为属性值,也就在找到bean中和Map中key相同的属性名,把这个key对应的value设置给这个属性。在程序上表达就是,提供一个方法,传入map对象和一个bean的Class类型,返回一个JavaBean对象。注意,这里传入的参数多了一个,因为肯定得用反射创建bean对象把,如果只传入一个map对象,怎么创建对象?怎么知道吧map转化为什么类型的JavaBean。好了,直接撸代码:

//把Map转化为JavaBean
public static <T> T map2bean(Map<String,Object> map,Class<T> clz) throws Exception{
    //创建一个需要转换为的类型的对象
    T obj = clz.newInstance();
    //从Map中获取和属性名称一样的值,把值设置给对象(setter方法)

    //得到属性的描述器
    BeanInfo b = Introspector.getBeanInfo(clz,Object.class);
    PropertyDescriptor[] pds = b.getPropertyDescriptors();
    for (PropertyDescriptor pd : pds) {
        //得到属性的setter方法
        Method setter = pd.getWriteMethod();
        //得到key名字和属性名字相同的value设置给属性
        setter.invoke(obj, map.get(pd.getName()));
    }
    return obj;
}
11-09 13:18