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

简介: 前几天接触到了apache的一个小框架DbUtils,真的被其优雅的设计所震撼到了,尤其是其中的MyBean mybean = QueryRunner.query(sqlConnection,sqlStatement,new BeanHandler(),params);当时真的是感觉到很是神奇,仅仅是指定了一下那个Bean类的全名,就能从数据库结果集中自动的生成我们需要的Bean对象,真的是太优雅了。

前几天接触到了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.Person
Person [name=person-name-property, age=20]
----------------------------------------
Qualify Name:com.grb.one.Dog
Dog [volun=dog-volun-property, name=dog-name-property]

总结

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

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

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

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

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

目录
相关文章
|
2月前
|
SQL XML Java
mybatis复习02,简单的增删改查,@Param注解多个参数,resultType与resultMap的区别,#{}预编译参数
文章介绍了MyBatis的简单增删改查操作,包括创建数据表、实体类、配置文件、Mapper接口及其XML文件,并解释了`#{}`预编译参数和`@Param`注解的使用。同时,还涵盖了resultType与resultMap的区别,并提供了完整的代码实例和测试用例。
mybatis复习02,简单的增删改查,@Param注解多个参数,resultType与resultMap的区别,#{}预编译参数
|
6月前
|
SQL XML Java
mybatis元素类型为 "resultMap" 的内容必须匹配 "(constructor?,id *,result*,association报错解决
mybatis元素类型为 "resultMap" 的内容必须匹配 "(constructor?,id *,result*,association报错解决
333 0
|
SQL 安全 Java
MyBatis动态语句且如何实现模糊查询及resultType与resultMap的区别---详细介绍
MyBatis动态语句且如何实现模糊查询及resultType与resultMap的区别---详细介绍
192 0
|
SQL Java 数据库连接
详解Mybatis查询之resultType返回值类型问题【4种情况】
详解Mybatis查询之resultType返回值类型问题【4种情况】
|
网络协议 Java 数据库连接
Springboot 通用返回类Result
Springboot 通用返回类Result
576 0
|
JSON JavaScript 数据格式
res 对象的常见方法|学习笔记
快速学习 res 对象的常见方法
res 对象的常见方法|学习笔记
|
Java 数据库连接
java使用jdbc连接ResultSet通过next()取不到第一条数据
java使用jdbc连接ResultSet通过next()取不到第一条数据
325 0
|
SQL Java 关系型数据库
记录:springboot使用resultType空指针java.lang.NullPointerException: null...【亲测有效】
记录:springboot使用resultType空指针java.lang.NullPointerException: null...【亲测有效】
434 0
|
JSON Java 数据库
Java数据库ResultSet转json实现
现在有很多json相关的Java工具,如json-lib、gson等,它们可以直接把JavaBean转换成json格式。 在开发中,可能会从数据库中获取数据,希望直接转成json数组,中间不通过bean。
2023 0