apache DbUtils 组件核心原理与应用

embedded/2024/10/18 21:43:13/

Apache DbUtils 是一个 Apache 组织提供的开源 JDBC 工具类库,它对 JDBC 进行了简单封装,使得数据库操作更加简洁和安全。DbUtils 的核心组件主要包括 QueryRunner、ResultSetHandler 和 RowProcessor,下面将对这些组件进行介绍,并结合源代码分析其工作原理。

核心组件介绍与原理

  1. QueryRunner:
  • QueryRunner 是执行数据库查询、更新和插入操作的入口类。
  • 它提供同步和异步服务,并且可以自动创建和管理 JDBC 资源(如 Connection、Statement、ResultSet)。
  • QueryRunner 还支持批处理操作,可以一次性执行多个 SQL 语句。
  1. ResultSetHandler:
  • ResultSetHandler 接口负责将 JDBC 的 ResultSet 转换成其他格式的数据。
  • DbUtils 提供了多种 ResultSetHandler 的实现,用于- 将结果集转换为数组、列表、Map、JavaBean 等格式。
  1. RowProcessor:
  • RowProcessor 用于处理行记录,将 ResultSet 的当前行转换为 Java 对象。
  • BasicRowProcessor 是 DbUtils 中的一个默认实现,用于将 SQL 行转换为 Map 或 JavaBean。

逻辑步骤:

  1. 创建 QueryRunner 实例:
  • 可以通过传递 DataSource 或 Connection 来创建 QueryRunner 对象。
  1. 编写 SQL 语句:
  • 准备要执行的 SQL 查询或更新语句。
  1. 选择 ResultSetHandler:
  • 根据需要处理的数据格式选择合适的 ResultSetHandler 实现。
  1. 执行查询或更新:
  • 使用 QueryRunner 的 query 或 update 方法执行 SQL 语句,并传入 ResultSetHandler。
  1. 处理结果:
  • ResultSetHandler 将 ResultSet 转换为相应的数据格式,并返回。
  1. 资源关闭:
  • DbUtils 提供了方法来安全关闭 JDBC 资源,避免资源泄漏。

源代码分析:

以 QueryRunner 和 BeanHandler 为例,分析其源码:

java">QueryRunner queryRunner = new QueryRunner();
BeanHandler<Emp> handler = new BeanHandler<>(Emp.class);
Emp emp = queryRunner.query("SELECT * FROM emp WHERE id = ?", handler, id);
  1. 创建 QueryRunner 对象:实例化 QueryRunner,准备执行数据库操作。

  2. 创建 BeanHandler 对象:BeanHandler 是 ResultSetHandler 的一个实现,用于将 ResultSet 的每一行转换为一个 JavaBean 对象。

  3. 执行查询:QueryRunner 的 query 方法接受 SQL 语句、ResultSetHandler 和参数,执行查询并返回结果。

QueryRunner 是 Apache DbUtils 中的一个核心类,它封装了 JDBC 的大部分操作,使得执行 SQL 语句更加方便和安全。以下是 QueryRunner 类的详细介绍和源码实现逻辑的分析:

QueryRunner 类介绍:

QueryRunner 类提供了执行各种 SQL 语句的方法,包括:

  • update:用于执行更新操作(如 INSERT、UPDATE、DELETE)。
  • query:用于执行查询操作并返回单个结果。
  • execute:用于执行 DDL 语句或存储过程调用。

它还支持批处理操作,可以一次性执行多个 SQL 语句。

源码实现逻辑:

QueryRunner 的实现涉及到几个关键的步骤:

  1. 获取 JDBC 连接:QueryRunner 的方法通常接收一个 Connection 对象作为参数。在执行 SQL 之前,你需要从数据源或连接池中获取一个 Connection。

  2. 准备 SQL 语句:QueryRunner 使用 PreparedStatement 来准备 SQL 语句。这有助于防止 SQL 注入攻击。

  3. 填充参数:如果 SQL 语句包含占位符(如 ?),QueryRunner 会根据提供的参数数组填充这些占位符。

  4. 执行 SQL:根据执行的 SQL 类型(更新或查询),QueryRunner 会调用适当的方法来执行 SQL。

  5. 处理结果:对于查询操作,QueryRunner 使用 ResultSetHandler 来处理 ResultSet 并将其转换为期望的结果类型。

  6. 资源管理:QueryRunner 提供了方法来确保所有 JDBC 资源(如 PreparedStatement 和 ResultSet)在使用后都被正确关闭,即使在发生异常的情况下也是如此。

源码分析:

以下是 QueryRunner 中 query 方法的一个简化示例,展示了其基本的实现逻辑:

java">public <T> T query(String sql, ResultSetHandler<T> rsh, Object... params) throws SQLException {Connection conn = null;PreparedStatement pstmt = null;ResultSet rs = null;try {conn = getConnection(); // 获取 JDBC 连接pstmt = conn.prepareStatement(sql);this.fillStatement(pstmt, params); // 填充参数rs = pstmt.executeQuery(); // 执行查询return rsh.handle(rs); // 使用 ResultSetHandler 处理结果集} finally {// 关闭资源,即使在发生异常的情况下JdbcUtils.close(rs);JdbcUtils.close(pstmt);JdbcUtils.close(conn);}
}

在这个简化的示例中,query 方法接收 SQL 语句、一个 ResultSetHandler 实例和一些参数。它使用 PreparedStatement 来执行 SQL 语句,并使用提供的 ResultSetHandler 来处理查询结果。

详细解释:

  • getConnection():这是一个从数据源或连接池中获取 Connection 的方法。在实际的 QueryRunner 实现中,这可以是传递给 QueryRunner 构造函数的 DataSource 或在方法中直接传递的 Connection。

  • fillStatement(pstmt, params):这是一个辅助方法,用于将参数绑定到 PreparedStatement 的占位符上。

  • rsh.handle(rs):这是 ResultSetHandler 接口的核心方法,它定义了如何处理 ResultSet。不同的 ResultSetHandler 实现会将 ResultSet 转换为不同的类型,如 JavaBean、Map、List 等。

  • finally 块:确保所有 JDBC 资源在使用后都被正确关闭。这是通过 JdbcUtils.close() 方法实现的,它是一个安全关闭资源的工具方法,可以处理 null 值和潜在的 SQLException。

QueryRunner 的设计哲学是简化 JDBC 编程,同时提供足够的灵活性来处理各种数据库操作。通过封装 JDBC 的复杂性,QueryRunner 使得开发者可以更专注于业务逻辑而不是底层的数据库操作细节。

应用操作

以下是一些 DbUtils 组件的应用案例,展示了如何使用 DbUtils 执行常见的数据库操作:

  1. 基本的数据库查询

使用 QueryRunner 和 BeanHandler 来执行查询并将结果映射到 JavaBean 对象。

java">QueryRunner queryRunner = new QueryRunner();
BeanHandler<User> handler = new BeanHandler<>(User.class);
User user = queryRunner.query(conn, "SELECT * FROM users WHERE id = ?", handler, 1);
  1. 查询并返回多条记录

使用 BeanListHandler 来处理多条记录,并将结果集映射为 JavaBean 对象的列表。

java">List<User> users = queryRunner.query(conn, "SELECT * FROM users", new BeanListHandler<>(User.class));
  1. 执行更新操作

使用 QueryRunner 的 update 方法来执行 INSERT、UPDATE 或 DELETE 语句。

java">int rowsInserted = queryRunner.update(conn, "INSERT INTO users(name, email) VALUES(?, ?)", "John Doe", "john@example.com");
  1. 批量更新

QueryRunner 支持批处理,可以一次性执行多个 SQL 更新操作。

java">Object[][] batchArgs = {{"John", "john@example.com"},{"Jane", "jane@example.com"}
};
queryRunner.batch(conn, "INSERT INTO users(name, email) VALUES(?, ?)", batchArgs);
  1. 处理聚集函数

使用 ScalarHandler 来处理返回单行单列的查询,如计数或总和。

java">Long count = queryRunner.query(conn, "SELECT COUNT(*) FROM users", new ScalarHandler<Long>(Long.class));
  1. 使用连接池

DbUtils 可以与连接池(如 DBCP 或 HikariCP)一起使用,以提高性能。

java">DataSource dataSource = ...; // 初始化数据源
QueryRunner queryRunner = new QueryRunner(dataSource);
// 使用 queryRunner 执行操作,不需要手动关闭连接
  1. 事务处理

DbUtils 提供了一些基本的事务管理方法,但通常与 Spring 或其他框架的事务管理集成更常见。

java">Connection conn = ...; // 获取数据库连接
try {conn.setAutoCommit(false); // 开始事务// 执行一系列数据库操作conn.commit(); // 提交事务
} catch (SQLException e) {conn.rollback(); // 回滚事务throw e;
} finally {conn.setAutoCommit(true); // 确保设置自动提交DbUtils.closeQuietly(conn); // 关闭连接
}
  1. 自定义 ResultSetHandler

如果预定义的 ResultSetHandler 实现不符合需求,可以自定义实现。

java">public class CustomHandler implements ResultSetHandler<List<MyBean>> {public List<MyBean> handle(ResultSet rs) throws SQLException {List<MyBean> list = new ArrayList<>();while (rs.next()) {// 根据需要从 ResultSet 中提取数据并创建 MyBean 对象MyBean bean = new MyBean();// ...list.add(bean);}return list;}
}

然后使用自定义的 ResultSetHandler:

java">List<MyBean> beans = queryRunner.query(conn, "SELECT * FROM my_table", new CustomHandler());

这些案例展示了 DbUtils 在实际应用中的灵活性和强大功能。通过这些组件,DbUtils 使得 JDBC 编程变得更加简洁和易于管理。

最后

DbUtils 的设计思想是简化 JDBC 编程,通过封装 JDBC 操作,减少样板代码,提高开发效率。它通过 QueryRunner、ResultSetHandler 和 RowProcessor 的协同工作,实现了对 JDBC 资源的精细化管理,同时避免了资源泄漏的风险。DbUtils 的使用不涉及复杂的配置和ORM映射,适合需要快速、轻量级数据库操作的场景。


http://www.ppmy.cn/embedded/26354.html

相关文章

FFMpeg - macOS build 报错 : xcrun -sdk iphoneos clang ...

文章目录 报错1&#xff1a;xcrun -sdk iphoneos clang is unable to create an executable file报错 2 &#xff1a; error: unknown type name AudioDeviceID; 在 macOS 上使用 https://github.com/kewlbear/FFmpeg-iOS-build-script 脚本&#xff0c;运行 ./build-ffmpeg.sh…

MCGS:脚本程序

MCGS仿真控制要求 控制要求如下 用PLC控制灯字闪灭 1、广告字1亮&#xff0c;1秒后熄灭&#xff1b; 2、广告字2亮&#xff0c;1秒后熄灭&#xff1b; 3、广告字3亮&#xff0c;1秒后熄灭&#xff1b; 4、广告字4亮&#xff0c;1秒后熄灭&#xff1b; 5、广告字5亮&#xff0c;…

spring boot运行过程中动态加载Controller

1.被加载的jar代码 package com.dl;import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication;SpringBootApplication public class App {public static void main(String[] args) {SpringApplication.run(A…

处理分支更新与pull操作

处理分支更新与pull操作 问题描述 There is no tracking information for the current branch. Please specify which branch you want to merge with. See git-pull(1) for details.git pull <remote> <branch> If you wish to set tracking information for th…

SpringCloudStream 3.x rabbit 使用

1. 前言 今天带来的是SpringCloudStream 3.x 的新玩法&#xff0c;通过四大函数式接口的方式进行数据的发送和监听。本文将通过 rabbitMQ 的方式进行演示 3.x版本后是 可以看到 StreamListener 和 EnableBinding 都打上了Deprecated 注解。后续的版本更新中会逐渐替换成函数式…

网络原理

端口号 IP地址&#xff08;确定主机&#xff09;端口号&#xff08;在机上的应用程序&#xff09; 网络层提供的概念 传输层提供的概念 一个字节&#xff1a; 有符号-128>127 无符号0>255 两个字节&#xff1a; 有符号&#xff1a;-3…

神经网络基础(Neural net foundations)

Today we’ll be learning about the mathematical foundations of deep learning: Stochastic gradient descent (SGD), and the flexibility of linear functions layered with non-linear activation functions. We’ll be focussing particularly on a popular combination…

基于H.264的RTP打包中的组合封包以及分片封包结构图简介及抓包分析;FU-A FU-B STAP-A STAP-B简介;

H.264视频流的RTP封装类型分析&#xff1a; 前言&#xff1a; 1.RTP打包原则&#xff1a; RTP的包长度必须要小于MTU(最大传输单元)&#xff0c;IP协议中MTU的最大长度为1500字节。除去IP报头&#xff08;20字节&#xff09;、UDP报头&#xff08;8字节&#xff09;、RTP头&a…