commons-dbutils工具介绍及实战

1.commons-dbutils简介

commons-dbutils 是 Apache 组织提供的一个开源 JDBC工具类库,它是对JDBC的简单封装,学习成本极低,并且使用dbutils能极大简化jdbc编码的工作量,同时也不会影响程序的性能。

API介绍:

•org.apache.commons.dbutils.QueryRunner

•org.apache.commons.dbutils.ResultSetHandler

•工具类

org.apache.commons.dbutils.DbUtils

链接如下:

https://commons.apache.org/proper/commons-dbutils/

2.DbUtils类介绍

DbUtils :提供如关闭连接、装载JDBC驱动程序等常规工作的工具类,里面的所有方法都是静态的(关闭资源、加载驱动)。主要方法如下:

•public static void close(…) throws java.sql.SQLException: DbUtils类提供了三个重载的关闭方法。这些方法检查所提供的参数是不是NULL,如果不是的话,它们就关闭Connection、Statement和ResultSet。

•public static void closeQuietly(…): 这一类方法不仅能在Connection、Statement和ResultSet为NULL情况下避免关闭,还能隐藏一些在程序中抛出的SQLException。

•public static void commitAndCloseQuietly(Connection

conn):

用来提交连接,然后关闭连接,并且在关闭连接时不抛出SQL异常。

•public static boolean loadDriver(java.lang.String driverClassName):这一方装载并注册JDBC驱动程序,如果成功就返回true。使用该方法,你不需要捕捉这个异常ClassNotFoundException。

3.QueryRunner类 介绍

该类简单化了SQL查询,它与ResultSetHandler组合在一起使用可以完成大部分的数据库操作,能够大大减少编码量。(核心工具类,定义了所有的与数据库操作的方法(查询、更新))

QueryRunner类提供了两个构造方法:

•默认的构造方法

需要一个 javax.sql.DataSource 来作参数的构造方法

QueryRunner类的主要方法

public Object query(Connection conn,String sql,Object[] params, ResultSetHandler rsh)

throws SQLException:执行一个查询操作,在这个查询中,对象数组中的每个元素值被用来作为查询语句的置换参数。该方法会自行处理 PreparedStatement和ResultSet的创建和关闭。

public Object query(String sql,Object[] params, ResultSetHandler rsh) throws SQLException: 几乎与第一种方法一样;唯一的不同在于它不将数据库连接提供给方法,并且它是从提供给构造方法的数据源(DataSource) 或使用的setDataSource 方法中重新获得 Connection。

public Object query(Connection conn,String sql, ResultSetHandler rsh) throws SQLException : 执行一个不需要置换参数的查询操作。

public int update(Connection conn, String sql,Object[] params) throws SQLException:用来执行一个更新(插入、更新或删除)操作。

public int update(Connection conn, String sql) throws SQLException:用来执行一个不需要置换参数的更新操作。

4.ResultSetHandler接口

该接口用于处理 java.sql.ResultSet,将数据按要求转换为另一种形式。

ResultSetHandler接口提供了一个单独的方法:Object

handle (java.sql.ResultSet .rs)。

ResultSetHandler 接口的实现类

ArrayHandler:把结果集中的第一行数据转成对象数组

ArrayListHandler:把结果集中的每一行数据都转成一个数组,再存放到List中。

BeanHandler:将结果集中的第一行数据封装到一个对应的JavaBean实例中。

BeanListHandler:将结果集中的每一行数据都封装到一个对应的JavaBean实例中,存放到List里。

ColumnListHandler(列名):将结果集中某一列的数据存放到List中。

KeyedHandler(name):将结果集中的每一行数据都封装到一个Map里,再把这些map再存到一个map里,其key为指定的key。

MapHandler:将结果集中的第一行数据封装到一个Map里,key是列名,value就是对应的值。

MapListHandler:将结果集中的每一行数据都封装到一个Map里,然后再存放到List

5.代码示例

1.JdbcUtils

package Utils;

import java.sql.*;

/**

* 1. 返回连接 2. 关闭

*

* @author dan.dan

*

*/

public class JdbcUtils {

// 连接参数

    // private String url = "jdbc:mysql://localhost:3306/jdbc_demo";

    private static Stringurl ="jdbc:mysql://10.9.1.43:3306/dandan_test";

private static Stringuser ="root";

private static Stringpassword ="123456";

/**

    * 返回连接对象

    */

    public static Connection getConnection() {

try {

Class.forName("com.mysql.jdbc.Driver");

return DriverManager.getConnection(url,user,password);

}catch (Exception e) {

throw new RuntimeException(e);

}

}

/**

    * 关闭

    */

    public static void closeAll(Connection con, Statement stmt, ResultSet rs) {

try {

if (rs !=null) {

rs.close();// 快速异常捕获Alt + shift + z

                rs =null;// 建议垃圾回收期回收资源

            }

if (stmt !=null) {

stmt.close();

stmt =null;

}

if (con !=null && !con.isClosed()) {

con.close();

con =null;

}

}catch (SQLException e) {

throw new RuntimeException(e);

}

}

}

package d_dbUtils;

public class Admin {

private int id;

private Stringuserid;

private Stringcn_score;

private Stringmath_score;

private Stringen_score;

private Stringpo_score;

public void setUserid(String userid) {

this.userid = userid;

}

public void setCn_score(String cn_score) {

this.cn_score = cn_score;

}

public void setMath_score(String math_score) {

this.math_score = math_score;

}

public void setEn_score(String en_score) {

this.en_score = en_score;

}

public void setPo_score(String po_score) {

this.po_score = po_score;

}

public String getUserid() {

return userid;

}

public String getCn_score() {

return cn_score;

}

public String getMath_score() {

return math_score;

}

public String getEn_score() {

return en_score;

}

public String getPo_score() {

return po_score;

}

public int getId() {

return id;

}

public void setId(int id) {

this.id = id;

}

@Override

    public String toString() {

return "Admin [id=" +id +", userid=" +userid +", " +

"cn_score=" +cn_score +"," +

"math_score=" +math_score +"," +

"en_score=" +en_score +"," +

"po_score=" +po_score +"]";

}

}

package d_dbUtils;

import Utils.JdbcUtils;

import org.apache.commons.dbutils.QueryRunner;

import org.apache.commons.dbutils.ResultSetHandler;

import org.apache.commons.dbutils.handlers.*;

import org.junit.Test;

import java.sql.Array;

import java.sql.Connection;

import java.sql.ResultSet;

import java.sql.SQLException;

import java.util.Arrays;

import java.util.List;

import java.util.Map;

import java.util.Set;

public class App_query {

private Connectionconn;

// 一、查询, 自定义结果集封装数据

    @Test

    public void testQuery()throws Exception {

String sql ="select * from tb_score1 where id=?";

// 获取连接

        conn = JdbcUtils.getConnection();

// 创建DbUtils核心工具类对象

        QueryRunner qr =new QueryRunner();

// 查询

        Admin admin = qr.query(conn, sql,new ResultSetHandler() {

// 如何封装一个Admin对象

            public Admin handle(ResultSet rs)throws SQLException {

if (rs.next()) {

Admin admin =new Admin();

admin.setId(rs.getInt("id"));

admin.setUserid(rs.getString("userid"));

admin.setCn_score(rs.getString("cn_score"));

return admin;

}

return null;

}

},29);

// 测试

      System.out.println(admin);

// 关闭

      conn.close();

}

// 二、查询, 使用组件提供的结果集对象封装数据

        // 1)BeanHandler: 查询返回单个对象

    @Test

    public void testQueryOne()throws Exception {

String sql ="select * from tb_score1 where id=?";

// 获取连接

        conn = JdbcUtils.getConnection();

// 创建DbUtils核心工具类对象

        QueryRunner qr =new QueryRunner();

// 查询返回单个对象

        Admin admin =  qr.query(conn, sql,new BeanHandler(Admin.class),30);

System.out.println(admin);

conn.close();

}

// 2)BeanListHandler: 查询返回list集合,集合元素是指定的对象

    @Test

    public void testQueryMany()throws Exception {

String sql ="select * from tb_score1";

conn = JdbcUtils.getConnection();

QueryRunner qr =new QueryRunner();

// 查询全部数据

        List list = qr.query(conn, sql,new BeanListHandler(Admin.class));

        conn.close();

}

@Test

// 3) ArrayHandler, 查询返回结果记录的第一行,封装对对象数组, 即返回:Object[]

// 4) ArrayListHandler, 把查询的每一行都封装为对象数组,再添加到list集合

// 5) ScalarHandler 查询返回结果记录的第一行的第一列  (在聚合函数统计的时候用)

// 6) MapHandler  查询返回结果的第一条记录封装为map

  public void testArray()throws Exception {

String sql ="select * from tb_score1";

conn = JdbcUtils.getConnection();

QueryRunner qr =new QueryRunner();

// 查询

      Object[] obj = qr.query(conn, sql,new ArrayHandler());

System.out.println("数组:"+ Arrays.toString(obj));

List list = qr.query(conn, sql,new ArrayListHandler());

for (int i =0; i < list.size(); i++) {

System.out.println("list:"+ Arrays.toString(list.get(i)));

} //查询的结果先是数组,然后将数组添加到list中,所以由list获取的每一条都为数组,然后遍历数组才能获取值

Integer num = qr.query(conn, sql,new ScalarHandler());

System.out.println("第一列:" + num);

Map map = qr.query(conn,sql,new MapHandler());

Set<Map.Entry<String, Object>> entrys = map.entrySet();

// entry代表一个键值对

        for (Map.Entry entry : entrys) {

System.out.println("map " + entry.getKey() +"=" + entry.getValue());

}

conn.close();

}

}

6.小结

使用dbutils简化了jdbc编码的工作量,不再用JDBC编写原子性代码,直接使用QueryRunner获取结果,而且其安全性较高,内部的JDBC一些工作已经做好了实现,用户只需直接使用即可。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,222评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,455评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,720评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,568评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,696评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,879评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,028评论 3 409
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,773评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,220评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,550评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,697评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,360评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,002评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,782评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,010评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,433评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,587评论 2 350

推荐阅读更多精彩内容