反射模拟DbUtils实现ResultSet转成Bean实例

来源:互联网 发布:mac 虚拟机免费 编辑:程序博客网 时间:2024/05/16 22:34

前几天接触到了apache的一个小框架DbUtils,真的被其优雅的设计所震撼到了,尤其是其中的

MyBean mybean = QueryRunner.query(sqlConnection,sqlStatement,new BeanHandler<MyBean.class>(),params);

当时真的是感觉到很是神奇,仅仅是指定了一下那个Bean类的全名,就能从数据库结果集中自动的生成我们需要的Bean对象,真的是太优雅了。然后我就翻了翻源码,然后尴尬的发现自己能力有限,并不能真正做出那样强大而且优雅的东西。

这里写图片描述


反射技术

虽然写不了那样强大的框架,但是模拟一下还是可行的嘛,我在源码中看到了反射技术的影子,然后就恶补了一下相关的知识点。细节方面主要是使用到了PropertyDescriptor这个类,以及相关的getReadMethodgetWriteMethod。说白了就是实现setter和getter方法。
下面是一个简单的入门级的小例子

// 这个小例子就包含了getter方法和setter的使用了public static <T> void test1(String proName, Class<T> beanCLass) throws Exception {        PropertyDescriptor pd = new PropertyDescriptor(proName, beanCLass);        String type = pd.getPropertyType().toString();        Method setMethod = pd.getWriteMethod();        // Person p = new Person();        T p = (T) beanCLass.newInstance();        // 这里也仅仅是做个示例,只允许String类型的Property来赋值,否则会报错的        setMethod.invoke(p, "郭璞");        System.out.println(pd.getReadMethod().invoke(p, null));    }

获取所有的Properties

经过了上面的小例子,那我们就具备了给单个的Property赋值的实现了,至于怎么给一个Bean的所有的Property赋值,请接着往下看。

beanClass.getDeclaredFields().有了这行代码,我们就可以接着下一步了。

// 使用下面的这个方法,我们就可以轻松的获取一个Bean的所有的Properties了public static <T> Field[] getFields(Class<T> clazz) throws Exception {        String qualifyName = new String(clazz.toString().substring(6, clazz.toString().length()));        qualifyName.concat(".class");        System.out.println("Qualify Name:" + qualifyName);        Class cls = Class.forName(qualifyName);        Field[] fields = cls.getDeclaredFields();        return fields;    }

我的小框架

为了接下来的测试成功,我们先new出来几个Bean吧。分别如下:

/** * @Date 2016年7月18日 * * @author Administrator */package com.grb.one;/** * @author 郭璞 * */public class Person {    private String name;    private int age;    public Person() {    }    public String getName() {        return name;    }    public void setName(String name) {        this.name = name;    }    public int getAge() {        return age;    }    public void setAge(int age) {        this.age = age;    }    public Person(String name) {        this.name = name;    }    @Override    public String toString() {        return "Person [name=" + name + ", age=" + age + "]";    }}/** * @Date 2016年7月18日 * * @author Administrator */package com.grb.one;/** * @author 郭璞 * */public class Dog {    public String volun;    private String name;    public String getVolun() {        return volun;    }    public void setVolun(String volun) {        this.volun = volun;    }    public Dog() {    }    @Override    public String toString() {        return "Dog [volun=" + volun + ", name=" + name + "]";    }    /**     * @param name     *            the name to set     */    public void setName(String name) {        this.name = name;    }    /**     * @return the name     */    public String getName() {        return name;    }    public void run(Integer mails) {        System.out.println("This dog runned:" + mails + " mails.!");    }}

接下来就是重头戏了。

/** * @Date 2016年7月18日 * * @author Administrator */package neixing;import java.beans.PropertyDescriptor;import java.lang.reflect.Field;import java.lang.reflect.Method;import java.util.HashMap;import java.util.Map;import com.grb.one.Dog;import com.grb.one.Person;/** * @author 郭璞 * */public class Demo {    public static void main(String[] args) throws Exception {        test1("name", com.grb.one.Person.class);        System.out.println("----------------------------------------");        test2();        System.out.println("----------------------------------------");        test3();    }    public static void test2() throws Exception {        Map<String, Object> resultset = new HashMap<String, Object>();        resultset.put("name", "person-name-property");        resultset.put("age", 20);        Person p = (Person) createBean(resultset, com.grb.one.Person.class);        System.out.println(p.toString());    }    public static void test3() throws Exception {        Map<String, Object> resultset = new HashMap<String, Object>();        resultset.put("volun", "dog-volun-property");        resultset.put("name", "dog-name-property");        Dog dog = (Dog) createBean(resultset, com.grb.one.Dog.class);        System.out.println(dog.toString());    }    public static <T> Field[] getFields(Class<T> clazz) throws Exception {        String qualifyName = new String(clazz.toString().substring(6, clazz.toString().length()));        qualifyName.concat(".class");        System.out.println("Qualify Name:" + qualifyName);        Class cls = Class.forName(qualifyName);        Field[] fields = cls.getDeclaredFields();        return fields;    }    public static <T> T createBean(Map<String, Object> resultset, Class<T> bean) throws Exception {        T mybean = bean.newInstance();        Field[] fields = getFields(bean);        for (Field field : fields) {            PropertyDescriptor pd = new PropertyDescriptor(field.getName(), bean);            Method setter = pd.getWriteMethod();            setter.invoke(mybean, resultset.get(field.getName()));        }        return mybean;    }}

测试结果

下面就来一起看看振奋人心的结果吧。

郭璞----------------------------------------Qualify Name:com.grb.one.PersonPerson [name=person-name-property, age=20]----------------------------------------Qualify Name:com.grb.one.DogDog [volun=dog-volun-property, name=dog-name-property]

总结

反射技术对于泛型的使用可谓是登峰造极了。这样做的好处不言而喻。
今天代码中的闪光点在于:

  • String.subString(); // 分割字符串,实现bean的全名转化成可反射的字符串类型

  • Map<String,Object>的使用,好处在于模拟了ResultSet,提供了素材

  • Class<T> bean的使用,神奇的一种方式啊。泛型的优点可不仅仅在这里能看到啊。:-)

好了,今天的分享就先到这里吧。希望这个思路能给看到这篇文章的童鞋一丝启发,开发出专属于自己的一套小工具。

0 0
原创粉丝点击