泛型的使用:模拟写出commons-dbutils

Posted xy-ouyang

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了泛型的使用:模拟写出commons-dbutils相关的知识,希望对你有一定的参考价值。

最近在复习泛型的知识,想起以前使用commons-dbutils的时候,觉得这个工具太厉害了。所以,试着自己瞎写看能不能模拟commons-dbutils的功能。

1、commons-dbutils的使用

  1.1、commons-dbutils是用来简化JDBC的代码。下面是其简单用法:

// 增删改
QueryRunner qr = new QueryRunner(JdbcUtils.getDataSource());// 创建QueryRunner,需要提供数据库连接池对象
String sql = "insert into t_students values(?,?,?,?)";// 给出sql模板
Object[] params = { 1, "liSi", 20, "female" };// 给出sql模板的参数
qr.update(sql, params);

// 查询
QueryRunner qr = new QueryRunner(JdbcUtils.getDataSource());
String sql = "select * from t_student where id = ?";
Object[] params = {1};
Stu stu = qr.query(sql, new BeanHandler<Stu>(Stu.class), params);
// 将结果集rs映射成javabean,要求结果集列名与javabean属性名一致

  1.2、commons-dbutils的其他查询用法

* BeanListHandler的应用,它是多行处理器
    - QueryRunner qr = new QueryRunner(JdbcUtils.getDataSource());
      String sql = "select * from t_stu";
      List<Stu> stuList = qr.query(sql,new BeanListHandler<Stu>(Stu.class));
* MapHandler的应用,它是单行处理器,把一行转换成一个Map对象
    - QueryRunner qr = new QueryRunner(JdbcUtils.getDataSource());
      String sql = "select * from t_stu where sid = ?";
      Object[] params = {1001};
      Map map = qr.query(sql,new MapHandler(), params);
* MapListHandler,它是多行处理器,把每行都转换成一个Map
    - QueryRunner qr = new QueryRunner(JdbcUtils.getDataSource());
      String sql = "select * from t_stu";
      List<Map<String,Object>> mapList = qr.query(sql, new MapListHandler());
* ScalarHandler的应用,它是单行单列时使用,最为合适
    - QueryRunner qr = new QueryRunner(JdbcUtils.getDataSource());
      String sql = "select count(*) from t_stu";
      Object obj = qr.query(sql,new ScalarHandler());

 

2、模拟commons-dbutils

技术图片

  2.1、工具类

  CommonUtils工具类的作用见我的博客:泛型的使用:封装工具类CommonUtils-把一个Map转换成指定类型的javabean对象(用到泛型)

  JdbcUtils类是用来获取数据库连接的,用到了c3p0连接池。

package com.oy.type;
import java.sql.Connection;
import java.sql.SQLException;
import javax.sql.DataSource;
import com.mchange.v2.c3p0.ComboPooledDataSource;

public class JdbcUtils {

    // 使用配置文件c3p0-config.xml, 放到src目录下
    private static ComboPooledDataSource dataSource = new ComboPooledDataSource();

    // 返回连接
    public static Connection getConnection(){
        try {
            return dataSource.getConnection();
        } catch (SQLException e) {
            e.printStackTrace();
        }
        return null;
    }

    // 返回连接池对象
    public static DataSource getDataSource() {
        return dataSource;
    }
}

 

  c3p0-config.xml:

<?xml version="1.0" encoding="UTF-8"?>
<c3p0-config>
    <default-config> 
        <property name="jdbcUrl">jdbc:mysql://localhost:3306/db_test?useUnicode=true&amp;characterEncoding=UTF-8</property>
        <property name="driverClass">com.mysql.jdbc.Driver</property>
        <property name="user">root</property>
        <property name="password"></property>
        <property name="acquireIncrement">3</property>
        <property name="initialPoolSize">10</property>
        <property name="minPoolSize">2</property>
        <property name="maxPoolSize">10</property>
    </default-config>
    <named-config name="name1">
        <property name="jdbcUrl">jdbc:mysql://localhost:3306/db_test</property>
        <property name="driverClass">com.mysql.jdbc.Driver</property>
        <property name="user">root</property>
        <property name="password">123</property>
        <property name="acquireIncrement">3</property>
        <property name="initialPoolSize">10</property>
        <property name="minPoolSize">2</property>
        <property name="maxPoolSize">10</property>
    </named-config>
</c3p0-config>

 

  2.2、QR类

package com.oy.type;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.List;
import javax.sql.DataSource;

public class QR<T> {
    private DataSource dataSource;

    public QR() {
        super();
    }

    public QR(DataSource dataSource) {
        super();
        this.dataSource = dataSource;
    }

    // 可以做增删改操作
    public int update(String sql, Object... params) {
        Connection con = null;
        PreparedStatement pstmt = null;
        try {
            con = dataSource.getConnection(); // 通过连接池得到连接对象
            pstmt = con.prepareStatement(sql);// 使用sql模板创建preparedStatement对象
            initParams(pstmt, params);// 对sql语句的?赋值
            return pstmt.executeUpdate(); // 执行
        } catch (Exception e) {
            throw new RuntimeException(e);
        } finally { // 释放资源
            try {
                if (pstmt != null) {
                    pstmt.close();
                }
                if (con != null) {
                    con.close();
                }
            } catch (SQLException e1) {
            }
        }
    }

    // 可以做查询操作
    public T query(String sql, BeanHandler<T> rh, Object... params) {
        Connection con = null;
        PreparedStatement pstmt = null;
        ResultSet rs = null;
        try {
            con = dataSource.getConnection(); // 通过连接池得到连接对象
            pstmt = con.prepareStatement(sql);// 使用sql模板创建preparedStatement对象
            initParams(pstmt, params);// 对sql语句的?赋值
            rs = pstmt.executeQuery(); // 执行查询,返回ResultSet对象

            return rh.handle(rs);// 传入结果集rs,得到T类型的对象
        } catch (Exception e) {
            throw new RuntimeException(e);
        } finally {
            try {
                if (rs != null) {
                    rs.close();
                }
                if (pstmt != null) {
                    pstmt.close();
                }
                if (con != null) {
                    con.close();
                }
            } catch (SQLException e) {
            }
        }
    }

    public List<T> query(String sql, BeanListHandler<T> rh, Object... params) {
        Connection con = null;
        PreparedStatement pstmt = null;
        ResultSet rs = null;
        try {
            con = dataSource.getConnection(); // 通过连接池得到连接对象
            pstmt = con.prepareStatement(sql);// 使用sql模板创建preparedStatement对象
            initParams(pstmt, params);// 对sql语句的?赋值
            rs = pstmt.executeQuery(); // 执行查询,返回ResultSet对象

            return rh.handle(rs);// 传入结果集rs,得到T类型的对象
        } catch (Exception e) {
            throw new RuntimeException(e);
        } finally {
            try {
                if (rs != null) {
                    rs.close();
                }
                if (pstmt != null) {
                    pstmt.close();
                }
                if (con != null) {
                    con.close();
                }
            } catch (SQLException e) {
            }
        }
    }

    // 给sql语句的?赋值
    private void initParams(PreparedStatement pstmt, Object... params) throws SQLException {
        if (params != null && params.length > 0) {
            for (int i = 0; i < params.length; i++) {
                pstmt.setObject(i + 1, params[i]);
            }
        }
    }
}

 

  2.3、BeanHandler:将查询的结果集转换成javabean

package com.oy.type;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.HashMap;
import java.util.Map;
public class BeanHandler<T> { private Class<T> clazz; public BeanHandler(Class<T> clazz) { this.clazz = clazz; } public T handle(ResultSet rs) throws SQLException { if (!rs.next()) return null; return CommonUtils.tobean(rsToMap(rs), clazz); } private Map<String, String> rsToMap(ResultSet rs) throws SQLException { Map<String, String> map = new HashMap<String, String>(); int count = rs.getMetaData().getColumnCount(); // 获取结果集的列数 for (int i = 1; i <= count; i++) { // 循环列 String columnName = rs.getMetaData().getColumnName(i);// 获取列名 String value = rs.getString(i); // 获取结果集一行中的每一列的值 map.put(columnName, value); } return map; } }

 

  2.4、BeanListHandler:将结果集转换成bean集合

package com.oy.type;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class BeanListHandler<T> {
    private Class<T> clazz;

    public BeanListHandler(Class<T> clazz) {
        this.clazz = clazz;
    }

    public List<T> handle(ResultSet rs) throws SQLException {
        List<T> list = new ArrayList<>();
        Map<String, String> map = null;

        while (rs.next()) { // 循环行
            map = new HashMap<String, String>();
            int count = rs.getMetaData().getColumnCount(); // 获取结果集的列数
            for (int i = 1; i <= count; i++) { // 循环列
                String columnName = rs.getMetaData().getColumnName(i);// 获取列名
                String value = rs.getString(i); // 获取结果集一行中的每一列的值
                map.put(columnName, value);
            }
            list.add(CommonUtils.tobean(map, clazz));
        }

        return list;
    }
}

 

  2.5、MapHandler:将结果集转换成Map

package com.oy.type;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.HashMap;
import java.util.Map;

public class MapHandler {

    public MapHandler() {}

    public Map<String, String> handle(ResultSet rs) throws SQLException {
        if (!rs.next())
            return null;
        return rsToMap(rs);
    }

    private Map<String, String> rsToMap(ResultSet rs) throws SQLException {
        Map<String, String> map = new HashMap<String, String>();
        int count = rs.getMetaData().getColumnCount(); // 获取结果集的列数
        for (int i = 1; i <= count; i++) { // 循环列
            String columnName = rs.getMetaData().getColumnName(i);// 获取列名
            String value = rs.getString(i); // 获取结果集一行中的每一列的值
            map.put(columnName, value);
        }
        return map;
    }
}

 

3、测试

package com.oy.type;
public class Stu {
    private Integer id;
    private Integer age;
    private String name;
    public Integer getId() {
        return id;
    }
    public void setId(Integer id) {
        this.id = id;
    }
    public Integer getAge() {
        return age;
    }
    public void setAge(Integer age) {
        this.age = age;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    @Override
    public String toString() {
        return "Stu [id=" + id + ", age=" + age + ", name=" + name + "]";
    }
    
}
package com.oy.type;
import java.util.List;
import org.junit.jupiter.api.Test;

public class StuDao {
    QR<Stu> qr = new QR<>(JdbcUtils.getDataSource());
    
    public Stu getStuById(Integer id) {
        String sql = "select * from stu where id = ?";
        Object[] params = {id};
        return qr.query(sql, new BeanHandler<Stu>(Stu.class), params);
    }
    
    public List<Stu> getStus(String ids) {
        String sql = "select * from stu where id in (" + ids + ")";
        return qr.query(sql, new BeanListHandler<Stu>(Stu.class));
    }
    
    @Test
    public void test1() {
        //System.out.println(getStuById(1));
        System.out.println(getStus("1,2"));
    }
}

 

以上是关于泛型的使用:模拟写出commons-dbutils的主要内容,如果未能解决你的问题,请参考以下文章

JSON反序列化泛型对象;泛型是变化的,如何写出通用代码?(源码分析)

JSON反序列化泛型对象;泛型是变化的,如何写出通用代码?(源码分析)

JSON反序列化泛型对象;泛型是变化的,如何写出通用代码?(源码分析)

Effective java -- 4 泛型

一段代码引发对泛型的重新认识

如何 EasyMock 对返回通配符泛型的方法的调用?