前言

关于Apache的DbUtils中间件或许了解的人并不多,大部分开发人员在生成环境中更 多的是依靠Hibernate、Ibatis、Spring JDBC、JPA等大厂提供的持久层技术解决方案,或者是企业内部自己研发的持久层技术。但无论如何,使用这些技术的初衷和本质都是为了能够减少企业开发 成本,提高生产效率,降低耦合。

放眼企业级项目,Hibernate等ORM产品是首选,而互联网领域,大部分开发人员往往并不会在生产环境中上这些ORM技术,原因很简单,要的就是效率,其次都不重要。对于刚接触SQL和JDBC的开发人员,最引以为傲的就是希望能够在日后编写复杂的SQL语句,以及会使用诸如Hibernate、Ibatis等第三方持久层技术,并且极力的撇清与传统JDBC技术的关系,但笔者不得不认为,这是一种普遍业界存在的“病态”!

如果是企业级的项目,尤其是跟金融相关的业务,SQL语句或许会非常复杂,并且关联着事物。 但互联网项目却并非如此,在互联网项目中,看你牛不牛逼并不是取决于你能否写出一条复杂的SQL语句,而是看你能否将原本一条复杂的SQL语句拆散成单条 SQL,一句一句的执行;并且脱离Hibernate等ORM产品后,能否使用传统的JDBC技术完成一条简单的CRUD操作,这才是牛逼!是的,你没有 听错,互联网确确实实就是这么玩还原最本质的东西,才是追求性能的不二选择

笔者本章不会提及垂直分库、水平分区等数据库概念,以及数据路由中间件等技术(请阅读笔者博 文《剖析淘宝TDDL—Matrix层分库分表实现》),因为这些内容与本章内容无关,但间接来看,笔者之前提及的单条SQL、使用JDBC完成基本的 CRUD操作就可以在最大程度上满足一个互联网场景的持久层操作。以Hibernate为例,简单来说需要经历HQL->SQL->DBMS 等编译过程,中间还冗余着缓存、对象等开销,希望大家记住,封装层次越高,性能越低!这个是无可争议的事实。笔者希望大家接下来,暂时“忘记”掉你所会的持久层技术,耐心的听笔者为你介绍Apache的DbUtils技术,或许你会有意想不到的收获。

目录

一、Apache Commons DbUtils简介;

二、下载与安装DbUtils;

三、使用DbUtils完成CRUD操作;

四、C3P0连接池集成DbUtils;

五、常用包、类讲解;

六、自动封装结果集;

七、事物管理;

一、Apache Commons DbUtils简介;

Apache的DbUtils工具是一个轻量级的持久层解决方案,天生为性能而生,它简单的对JDBC进行了必要的操作封装,让开发人员能够以一种高级API的方式使用JDBC技术完成原本复杂的CRUD操作。换句话说,DbUtils天生就不是一个复杂的技术,它只是一个简单的JDBC上层封装, 对开发人员而言,大概只需半小时就能够完全掌握DbUtils技术的使用,是的,它就是这么简单与方便,它是互联网项目的宠儿,选择DbUtils技术作 为持久层的解决方案,或许能够让你从原本复杂的Hibernate操作中解脱出来,或者是你觉得Ibatis不够好用,DbUtils也是你选择的理由之 一。总之,使用它,你将会感到惊艳,它是如此的简单和干净,如此的纯粹和高效!并且DbUtils是采用商业友好的开源协议,大家甚至可以下载它的源码, 进行二次开发,以此满足企业自身的需要。

二、下载与安装DbUtils;

当大家对DbUtils的项目背景有所了解后,接下来本节内容笔者将会告诉你它的下载和安装。大家可以登录http://commons.apache.org/站点下载DbUtils工具的最新版本,笔者使用的版本为1.6.0,在此大家需要注意,为了避免在开发过程中出现异常,建议大家下载、使用与笔者本篇博文一致的版本。

当大家成功下载好DbUtils相关的构件后,我们可以将其添加到项目中的 ClassPath目录下,当然笔者后续小节会提及DbUtils与C3P0连接池的集成,因此,大家最好将C3P0所需的构件以及数据库驱动(笔者使用 Mysql)一起添加到项目中。使用DbUtils时关联的构件,如下所示:

3、使用DbUtils完成CRUD操作;

废话不多说,使用DbUtils操作数据库之前,首先要做的事情就是获取Connection。那么为了方便,笔者使用硬编码的方式将数据源的配置信息coding在代码中(生产环境中,有可能是配置在项目的配置文件中、数据库中、Diamond中等),如下所示:

  1. /**
  2. * 数据源信息
  3. *
  4. * @author gaoxianglong
  5. */
  6. public class ConnectionManager {
  7. public static Connection getConnection() {
  8. Connection conn = null;
  9. try {
  10. Class.forName("com.mysql.jdbc.Driver");
  11. conn = DriverManager.getConnection(
  12. "jdbc:mysql://ip:port/dbName", "userName",
  13. "passWord");
  14. } catch (Exception e) {
  15. e.printStackTrace();
  16. }
  17. return conn;
  18. }
  19. }

当编写好ConnectionManager之后,接下来要做的事情就是获取 Connection,然后就能够使用DbUtils进行CRUD操作了。或许细心的读者已经发现,使用DbUtils其实是非常简单的,需要会的不多, 仅仅只需要掌握JDBC操作以及简单的CRUD操作即可(互联网场景下同样也是这么要求)。

  1. @Test
  2. public void testInsert() {
  3. final String SQL = "insert into test_1 values(?, ?)";
  4. try {
  5. if (null == conn || conn.isClosed())
  6. conn = ConnectionManager.getConnection2();
  7. int result = new QueryRunner().update(conn, SQL, new Object[] {
  8. "JohnGao1", "123" });
  9. if (0 < result)
  10. System.out.println("数据插入成功...");
  11. } catch (Exception e) {
  12. e.printStackTrace();
  13. } finally {
  14. close(conn);
  15. }
  16. }
  17. @Test
  18. public void testUpdate() {
  19. final String SQL = "update test_1 set password= ? where username = ?";
  20. try {
  21. if (null == conn || conn.isClosed())
  22. conn = ConnectionManager.getConnection();
  23. int result = new QueryRunner().update(conn, SQL, new Object[] {
  24. "321", "JohnGao1" });
  25. if (0 < result)
  26. System.out.println("数据更新成功...");
  27. } catch (Exception e) {
  28. e.printStackTrace();
  29. } finally {
  30. close(conn);
  31. }
  32. }
  33. @Test
  34. public void testDelete() {
  35. final String SQL = "delete from test_1 where username like ?";
  36. try {
  37. if (null == conn || conn.isClosed())
  38. conn = ConnectionManager.getConnection();
  39. int result = new QueryRunner().update(conn, SQL, "%JohnGao%");
  40. if (0 < result)
  41. System.out.println("数据删除成功...");
  42. } catch (Exception e) {
  43. e.printStackTrace();
  44. } finally {
  45. close(conn);
  46. }
  47. }
  48. @Test
  49. public void testQuery() {
  50. final String SQL = "select * from test_1";
  51. try {
  52. if (null == conn || conn.isClosed())
  53. conn = ConnectionManager.getConnection();
  54. Test_1Bean test1Bean = new QueryRunner().query(conn, SQL,
  55. new BeanHandler(Test_1Bean.class));
  56. if (null != test1Bean) {
  57. System.out.println(test1Bean.getUsername());
  58. System.out.println(test1Bean.getPassword());
  59. }
  60. } catch (Exception e) {
  61. e.printStackTrace();
  62. } finally {
  63. close(conn);
  64. }
  65. }

四、C3P0连接池集成DbUtils;

在生产环境中,开发人员在对数据库进行CRUD操作的时候,由于数据库的链接是有限的,因此不得不使用连接池来实现资源复用,以此降低数据库的性能瓶颈(尽管这么说有些不太友好,因为并发环境下,单靠连接池是不能够解决问题的,而常用的方案更多是诸如Redis之类的内存数据库抗住70%传统DBMS数据的受访压力、数据库先做垂直分库,再做水平分区,当然Master/Sleave是必不可少的,经过这些步骤之后,才能够说基本上解决了理论上可能出现的数据库在高并发环境下的瓶颈)。

废话不多说,在之前的ConnectionManager中添加进连接池相关的代码,当然为了方便,笔者同样还是使用硬编码的方式,如下所示:

  1. public static ComboPooledDataSource dataSource;
  2. static {
  3. try {
  4. dataSource = new ComboPooledDataSource();
  5. dataSource.setUser("userName");
  6. dataSource.setPassword("passWord");
  7. dataSource.setJdbcUrl("jdbc:mysql://ip:port/dbName");
  8. dataSource.setDriverClass("com.mysql.jdbc.Driver");
  9. dataSource.setInitialPoolSize(10);
  10. dataSource.setMinPoolSize(5);
  11. dataSource.setMaxPoolSize(50);
  12. dataSource.setMaxStatements(100);
  13. dataSource.setMaxIdleTime(60);
  14. } catch (Exception e) {
  15. e.printStackTrace();
  16. }
  17. }
  18. /**
  19. * 从连接池中获取数据源链接
  20. *
  21. * @author gaoxianglong
  22. *
  23. * @return Connection 数据源链接
  24. */
  25. public static Connection getConnection2() {
  26. Connection conn = null;
  27. if (null != dataSource) {
  28. try {
  29. conn = dataSource.getConnection();
  30. } catch (SQLException e) {
  31. e.printStackTrace();
  32. }
  33. }
  34. return conn;
  35. }

当成功在ConnectionManager中添加好所需的C3P0连接池配置后,接下来要做的事情就是考虑如何使用C3P0与DbUtils之间的集成。其实最简单的做法就 是直接将之前获取Connection的getConnection()方法更换为上述代码中的getConnection2()即可,同样可以使用在创 建QueryRunner实例时,将数据源的DataSource传递过去,这样即可避免在执行CRUD操作时,还需要在方法中指明 Connection。

当然究竟应该怎么做,完全取决于你自己,如果希望方便,那么笔者建议你在创建QueryRunner实例时,直接将C3P0的DataSource传递过去,但这样做的弊端很明显,如果在特殊的场景下,需要手动控制事物时,那么这种操作是极其不便的,因为Connection并不可控。那么为了解决事物控制的问题,当然是Connection可控最好。

五、常用包、类讲解;

相信大家已经从上述DbUtils的CRUD示例中发现了QueryRunner的身影,那么笔者接下来就将会针对DbUtils中诸如QueryRunner等常用类型进行深入讲解。

在DbUtils中,最常用的3个包为org.apache.commons.dbutils、org.apache.commons.dbutils.handlers以及org.apache.commons.dbutils.wrappers。

org.apache.commons.dbutils包下的常用类,如下所示:

1、DbUtils : 提供如关闭连接、装载 JDBC 驱动程序等常规工作的工具类;
2、QueryRunner : 该类简单化了 SQL 查询,它常与与 ResultSetHandler 组合在一起使用;

org.apache.commons.dbutils.handlers包下的常用类,如下所示:

1、ArrayHandler :将ResultSet中第一行的数据转化成对象数组;
2、ArrayListHandler:将ResultSet中所有的数据转化成List,List中存放的是Object[];
3、BeanHandler :将ResultSet中第一行的数据转化成类对象;
4、BeanListHandler :将ResultSet中所有的数据转化成List,List中存放的是类对象;
5、ColumnListHandler :将ResultSet中某一列的数据存成List,List中存放的是Object对象;
6、KeyedHandler :将ResultSet中存成映射,key为某一列对应为Map。Map中存放的是数据;
7、MapHandler :将ResultSet中第一行的数据存成Map映射;
8、MapListHandler :将ResultSet中所有的数据存成List。List中存放的是Map;
9、ScalarHandler :将ResultSet中一条记录的其中某一列的数据存成Object;

org.apache.commons.dbutils.wrappers包下的常用类,如下所示:

1、SqlNullCheckedResultSet :该类是用来对sql语句执行完成之后的的数值进行null的替换;
2、StringTrimmedResultSet :去除ResultSet中中字段的左右空格;

六、自动封装结果集;

在org.apache.commons.dbutils.handlers包下的类型,大部分都是与查询结果集相关的。试想一下,利用传统的JDBC进行查询时,返回的数据我们需要对ResultSet进行迭代,这是相当麻烦的,且不利于维护,因为我们需要手动编写与之相关的数据封装。但是使用DbUtils之后,我们要做的事情仅仅只是告诉DbUtils我们需要什么样的数据即可,关于数据封装这种通用的控制逻辑,则无需开发人员参与,这极大的节省了开发人员的时间,提升了生产效率。

简单来说,笔者在开发过程中使用最广泛的就是BeanListHandler以及
MapListHandler
封装的结果集。简单来说,BeanListHandler将会查询后的数据封装到一个对应的POJO中(可以看做是一个无状态的实体
Bean),MapListHandler
会将查询后的数据封装为一个List,List中存储的就是一个个的Map集合,通过key-value的方式获取封装后的数据集。先来看看
MapListHandler 的使用,如下所示:

  1. @Test
  2. public void testQuery4() {
  3. final String SQL = "select * from test_1 where username like ?";
  4. try {
  5. if (null == conn || conn.isClosed())
  6. conn = ConnectionManager.getConnection2();
  7. List<Map<String, Object>> values = new QueryRunner().query(conn,
  8. SQL, new Object[] { "%JohnGao%" }, new MapListHandler());
  9. if (null != values) {
  10. for (int i = 0; i < values.size(); i++) {
  11. Map<String, Object> map = values.get(i);
  12. System.out.println(map.get("username"));
  13. System.out.println(map.get("password"));
  14. }
  15. }
  16. } catch (Exception e) {
  17. e.printStackTrace();
  18. } finally {
  19. close(conn);
  20. }
  21. }

如果你喜欢类似于实体Bean的操作方式,那么BeanListHandler无疑使最好的
选择。一旦我们使用BeanListHandler作为数据返回后的结果集封装,那么DbUtils便会将查询后的结果集一个字段一个字段的映射到指定的
POJO中,当然前提就是字段名称是必须一致的,否则DbUtils将无法完成数据封装。BeanListHandler的使用示例,如下所示:

  1. @Test
  2. public void testQuery3() {
  3. final String SQL = "select * from test_1 where username like ?";
  4. try {
  5. if (null == conn || conn.isClosed())
  6. conn = ConnectionManager.getConnection();
  7. List<Test_1Bean> test1Beans = new QueryRunner().query(conn, SQL,
  8. new Object[] { "%JohnGao%" }, new BeanListHandler(
  9. Test_1Bean.class));
  10. if (null != test1Beans) {
  11. for (Test_1Bean test1Bean : test1Beans) {
  12. System.out.println(test1Bean.getUsername());
  13. System.out.println(test1Bean.getPassword());
  14. }
  15. }
  16. } catch (Exception e) {
  17. e.printStackTrace();
  18. } finally {
  19. close(conn);
  20. }
  21. }

在此大家需要注意,为了方便演示,笔者在此并没有提供对应的POJO。如果有需要,大家可以编写一个与数据库表字段相同的POJO来完成查询结果集的字段映射封装操作。

七、事物管理;

说起事物管理,这其实是一个非常复杂与繁琐,且是最容易出错的场景,尤其是在手动管理事物操
作上。当然本节所提及的事物管理仍然是建立在基于手动管理的事物操作上。对于JDBC操作,如果希望事物不要手动提交,那么在获取Connection的
时候,一定需要将设置conn.setAutoCommit(false);这样一来事物就不会自动进行提交,当我们手动执行conn.commit()
方法的时候,事物才会进行提交。这种方式对于DbUtils其实是一样的,之前也说过,DbUtils仅仅只是对JDBC做了一个轻量级的上层封装,那么
必然可以和JDBC进行混用,一旦我们在程序中设定了事物后,接下来的事物管理操作就依赖与开发人员自身了,DbUtils将不会再参与事物的管理。

对于大多数开发人员而言,事物控制的不好,将会导致业务出现问题,脏数据等情况是非常常见的,但从另一个层面来说,手动的事物管理其实是最灵活和方便的。在此需要提醒大家,如果是使用Mysql数据库,只有将数据库引擎设置为InnoDB后,才会支持事物

最后笔者在啰嗦一下,使用完资源后,我们一定要记得及时释放掉资源,以此避免无用资源长时间
挂起。那么在DbUtils中,你将有2种方式结束掉Connection,第一个是使用DbUtils.close()方法。其次,你将可以直接使用
close()方法关闭Connection的链接。

《笔者带你剖析Apache Commons DbUtils 1.6》(转)的更多相关文章

  1. 转《在浏览器中使用tensorflow.js进行人脸识别的JavaScript API》

    作者 | Vincent Mühle 编译 | 姗姗 出品 | 人工智能头条(公众号ID:AI_Thinker) [导读]随着深度学习方法的应用,浏览器调用人脸识别技术已经得到了更广泛的应用与提升.在 ...

  2. face-api.js:一个在浏览器中进行人脸识别的 JavaScript 接口

    Mark! 本文将为大家介绍一个建立在「tensorflow.js」内核上的 javascript API——「face-api.js」,它实现了三种卷积神经网络架构,用于完成人脸检测.识别和特征点检 ...

  3. TensorFlow.js之安装与核心概念

    TensorFlow.js是通过WebGL加速.基于浏览器的机器学习js框架.通过tensorflow.js,我们可以在浏览器中开发机器学习.运行现有的模型或者重新训练现有的模型. 一.安装     ...

  4. 在Java中直接调用js代码(转载)

    http://blog.csdn.net/xzyxuanyuan/article/details/8062887 JDK1.6版添加了新的ScriptEngine类,允许用户直接执行js代码. 在Ja ...

  5. 第十一章:WEB浏览器中的javascript

    客户端javascript涵盖在本系列的第二部分第10章,主要讲解javascript是如何在web浏览器中实现的,这些章节介绍了大量的脚本宿主对象,这些对象可以表示浏览器窗口.文档树的内容.这些章节 ...

  6. 在Java中直接调用js代码

    JDK1.6版添加了新的ScriptEngine类,允许用户直接执行js代码. 在Java中直接调用js代码 不能调用浏览器中定义的js函数,会抛出异常提示ReferenceError: “alert ...

  7. TensorFlow.js入门(一)一维向量的学习

    TensorFlow的介绍   TensorFlow是谷歌基于DistBelief进行研发的第二代人工智能学习系统,其命名来源于本身的运行原理.Tensor(张量)意味着N维数组,Flow(流)意味着 ...

  8. JavaScript权威指南--WEB浏览器中的javascript

    知识要点 1.客户端javascript window对象是所有客户端javascript特性和API的主要接入点.它表示web浏览器的一个窗口或窗体,并且可以用window表示来引用它.window ...

  9. 解决webkit浏览器中js方法中使用window.event提示未定义的问题

    这实际上是一个浏览器兼容性问题,根源百度中一大堆,简要说就是ie中event对象是全局变量,所以哪里都能使用到,但是webkit内核的浏览器中却不存在这个全局变量event,而是以一个隐式的局部变量的 ...

  10. JS Date当前时间:获取日期时间方法在各浏览器中的差异

    转自:http://www.feiesoft.com/00047/<script type="text/javascript"> // JS Date当前时间获取方法在 ...

随机推荐

  1. <nginx+PHP>nginx环境下配置支持php7

    [root@redhat7 ~]# wget http://am1.php.net/get/php-7.1.2.tar.gz/from/this/mirror [root@redhat7 ~]# ta ...

  2. java.lang.ClassNotFoundException: org.apache.commons.discovery.tools.DiscoverSingleton

    java.lang.ClassNotFoundException: org.apache.commons.discovery.tools.DiscoverSingleton org.apache.ax ...

  3. 华为S5700系列交换机AR配置静态IP双链路负载分担

    适用于:有多个以太WAN口的机型. 业务需求: 运营商1分配的接口IP为100.100.1.2,子网掩码为255.255.255.252,网关IP为100.100.1.1. 运营商2分配的接口IP为2 ...

  4. oracle 10g和11g将表到缓存到内存中

    alter table 表名 cache;alter table 表名 storage(buffer_pool keep);

  5. Java中的条件运算符

    条件运算符( ? : )也称为 “三元运算符”. 语法形式:布尔表达式 ? 表达式1 :表达式2 运算过程:如果布尔表达式的值为 true ,则返回 表达式1 的值,否则返回 表达式2 的值 例如: ...

  6. Minhash 算法 及其应用

    背景: 我遇到一个问题,要计算140万商品的杰卡德相似度.如果直接要直接两两计算的话,这计算量根本算不了,而且也没必要. 分析: 在这些商品中很多商品的相似度并不高,也就是说其中达到相似度阈值的商品只 ...

  7. 绝对布局absoluteLayout

    绝对布局absoluteLayout 一.简介 二.实例 绝对布局我们是指定的横纵坐标,所以可以这样直接拖 绝对布局实际中用的少

  8. 解决:actual_tessdata_num_entries_ <= TESSDATA_NUM_ENTRIES:Error:Assert failed:in file ..\..\ccutil\tessdatamanager.cp p, line 50

    在玩tesseract时,发现如下报错: 这个是因为Tesseract-OCR的版本和chi_sim.traindata字库版本不匹配,由于我的Tesseract-OCR是3.02.02,去googl ...

  9. ultraedit使用记录

    ultraedit使用记录 10:57:33 在日常的工作中,我经常用keil进行程序的编写等工作,不过在编写过程中Keil对中文的支持不是很好,容易发生问题:同事推荐我用ultraedit进行程序的 ...

  10. 二 web爬虫,scrapy模块以及相关依赖模块安装

    当前环境python3.5 ,windows10系统 Linux系统安装 在线安装,会自动安装scrapy模块以及相关依赖模块 pip install Scrapy 手动源码安装,比较麻烦要自己手动安 ...