JDBC ResultSet

2024-06-04 19:52
文章标签 jdbc resultset

本文主要是介绍JDBC ResultSet,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

JDBC ResultSet

  • ResultSet包含记录

  • 创建ResultSet

    • ResultSet类型、并发性和可保持性
  • 迭代ResultSet

  • 访问列值

  • ResultSet类型

    • 导航方法
  • ResultSet并发

  • 更新ResultSet

  • 将行插入ResultSet中

  • ResultSet可保持性

Java JDBC ResultSet接口表示数据库查询的结果。关于查询的文本显示了如何将查询的结果作为java.sql.ResultSet返回。然后迭代此ResultSet以检查结果。本教程介绍如何使用ResultSet接口。

结果集包含记录

JDBC ResultSet包含记录。每条记录都包含一组列。每个记录包含相同数量的列,尽管并非所有列都有一个值。列可以具有null值。以下是JDBC ResultSet的示例:

NameAge
张三22
李四23
王五25

ResultSet示例-带列的记录

ResultSet有2个不同的列(Name、Age),每列有3个具有不同值的记录。

创建结果集

通过执行StatementPreparedStatement创建ResultSet,如下所示:

Statement statement = connection.createStatement();ResultSet result = statement.executeQuery("select * from test");

或者像这样:

String sql = "select * from test";
PreparedStatement statement = connection.prepareStatement(sql);ResultSet result = statement.executeQuery();

ResultSet类型、并发性和可保持性

创建 ResultSet时,可以设置三个属性。这些是:

  1. 类型
  2. 并发
  3. 可持有性

您在创建StatementPreparedStatement时已经设置了这些,如下所示:

Statement statement = connection.createStatement(ResultSet.TYPE_FORWARD_ONLY,ResultSet.CONCUR_READ_ONLY,ResultSet.CLOSE_CURSORS_OVER_COMMIT);PreparedStatement statement = connection.prepareStatement(sql,ResultSet.TYPE_FORWARD_ONLY,ResultSet.CONCUR_READ_ONLY,ResultSet.CLOSE_CURSORS_OVER_COMMIT);

这些属性的确切含义将在本文后面解释。

迭代ResultSet

要迭代ResultSet,可以使用它的next()方法。如果ResultSet有下一条记录,则next()方法返回true,并将ResultSet移动到下一个记录。如果没有更多的记录,next()将返回false,您就不能再这样做了。一旦next()方法返回false,就不应该再调用它了。否则这样做可能会导致异常。

以下是使用next()方法迭代ResultSet的示例:

while(result.next()) {// result.getString("name")
}

正如您所看到的,next()方法实际上是在访问第一条记录之前调用的。这意味着ResultSet开始指向第一条记录之前。一旦next()被调用一次,它就会指向第一条记录。

类似地,当next()被调用并返回false时,ResultSet实际上指向最后一条记录之后。

除非您对ResultSet中的所有行进行迭代并对行数进行计数,否则无法获得结果集中的行数。但是,如果ResultSet仅向前,则以后不能向后移动。即使可以向后移动,这也是计算ResultSet中行数的缓慢方法。你最好构建你的代码,这样你就不需要提前知道记录的数量。

访问列值

迭代ResultSet时,您希望访问每条记录的列值。您可以通过调用许多getXXX()方法中的一个或多个来实现这一点。将要获取的值的列的名称传递给许多getXXX()方法。例如:

while(result.next()) {result.getString    ("name");result.getInt       ("age");result.getBigDecimal("money");}

你可以调用很多getXXX()方法,它们以某种数据类型返回列的值,例如String、int、long、double、BigDecimal等。它们都以列的名称来获取的列值作为参数。以下是这些getXXX()方法的快速示例列表:

result.getArray("columnName");
result.getAsciiStream("columnName");
result.getBigDecimal("columnName");
result.getBinaryStream("columnName");
result.getBlob("columnName");
result.getBoolean("columnName");
result.getBlob("columnName");
result.getBoolean("columnName");
result.getByte("columnName");
result.getBytes("columnName");
result.getCharacterStream("columnName");
result.getClob("columnName");
result.getDate("columnName");
result.getDouble("columnName");
result.getFloat("columnName");
result.getInt("columnName");
result.getLong("columnName");
result.getNCharacterStream("columnName");
result.getObject("columnName");
result.getRef("columnName");
result.getRowId("columnName");
result.getShort("columnName");
result.getSQLXML("columnName");
result.getString("columnName");
result.getTime("columnName");
result.getTimestamp("columnName");
result.getUnicodeStream("columnName");
result.getURL("columnName");

getXXX()方法也有采用列索引而不是列名的版本。例如:

while(result.next()) {result.getString    (1);result.getInt       (2);result.getBigDecimal(3);
}

列的索引通常取决于SQL语句中该列的索引。例如,SQL语句

select name, age, money from test

有三列。name列在第一位,因此在ResultSet中将具有索引1。age列将具有索引2,money列将具有索引3。

有时你不知道某一列的索引提前。例如,如果使用select * from test类型的SQL查询,则不知道列的顺序。
如果你不知道某一列的索引,你可以使用ResultSet.findColum(StringcolumnName)方法找到该列的索引

int nameIndex   = result.findColumn("name");
int ageIndex    = result.findColumn("age");
int coeffIndex  = result.findColumn("money");while(result.next()) {String     name        = result.getString(nameIndex);int        age         = result.getInt(ageIndex);BigDecimal money       = result.getBigDecimal(money);
}

ResultSet类型

ResultSet可以是特定类型的。类型决定了ResultSet的一些特性和能力。
并非所有数据库和JDBC驱动程序都支持所有类型。您必须检查数据库和JDBC驱动程序,看看它是否支持您想要使用的类型。DatabaseMetaData.supportsResultSetType(int类型)方法返回truefalse,具体取决于是否支持给定的类型。DatabaseMetaData类将在后面的文本中介绍。

就目前我知道的,有三种ResultSet类型:

  1. ResultSet.TYPE_FORWARD_ONLY
  2. ResultSet.TYPE_SCROLL_INSENSITIVE
  3. ResultSet.TYPE_SCROLL_SENSITIVE

默认类型为TYPE_FORWARD_ONLY

TYPE_FORWARD_ONLY表示只能向前导航ResultSet。也就是说,您只能从第1行移动到第2行、第3行等。您不能在ResultSet中向后移动。

TYPE_SCROLL_INSENSITIVE表示可以向前和向后导航(滚动)ResultSet。也可以跳到相对于当前位置的位置,或者跳到绝对位置。当ResultSet处于打开状态时,ResultSet对基础数据源中的更改不敏感。也就是说,如果另一个线程或进程在数据库中更改了ResultSet中的记录,它将不会反映在已打开的此类ResultSet中。

TYPE_SCROLL_SENSITIVE表示可以向前和向后导航(滚动)ResultSet。也可以跳到相对于当前位置的位置,或者跳到绝对位置。当ResultSet处于打开状态时,ResultSet对基础数据源中的更改很敏感。也就是说,如果数据库中的ResultSet中的记录被另一个线程或进程更改,它将反映在已打开的此类ResultSet中。

导航方法

ResultSet接口包含以下导航方法。请记住,并非所有方法都适用于所有ResultSet类型。什么方法有效取决于您的数据库、JDBC驱动程序和ResultSet类型。

方法描述
absolute()将ResultSet移动到绝对位置。位置是作为参数传递给absolute()方法的行号
afterLast()将ResultSet移动到ResultSet中最后一行的后面
beforeFirst()将ResultSet移动到ResultSet的第一行之前
first()将ResultSet移动到ResultSet中的第一行
last()将ResultSet移动到ResultSet中的最后一行
next()将ResultSet移动到ResultSet中的下一行
previous()将结果集移动到结果集中的前一行
relative()将ResultSet移动到相对于其当前位置的位置。相对位置作为参数传递给相对方法,可以是正的也可以是负的
移动结果集

ResultSet接口还包含一组方法,可用于查询ResultSet的当前位置。这些是:

方法描述
getRow()返回当前行的行号-ResultSet当前指向的行
getType()返回ResultSet类型
isAfterLast()如果ResultSet指向最后一行之后,则isAfterLast()返回true。否则为False
isBeforeFirst()如果ResultSet指向最后一行之后,则返回true。否则为False
isFirst()如果ResultSet指向第一行之前,则返回true。否则为False

ResultSet接口还包含一个方法,用于在ResultSet对更改敏感的情况下,通过数据库更改来更新行。

方法描述
refreshRow()使用数据库中的最新值刷新该行的列值

ResultSet并发

ResultSet并发性决定了ResultSet是可以更新还是只能读取。

某些数据库和JDBC驱动程序支持更新ResultSet,但并非所有数据库和JDBC驱动器都支持。DatabaseMetaData.supportsResultSetConcurrency(intconcurrency)方法返回truefalse,具体取决于是否支持给定的并发模式。DatabaseMetaData类将在后面进行介绍。

ResultSet可以具有以下两个并发级别之一:

  1. ResultSet.CONCUR_READ_ONLY
  2. ResultSet.CONCUR_UPDABLE

CONCUR_READ_ONLY表示只能读取ResultSet
CONCUR_UPDABLE表示ResultSet既可以读取也可以更新。

更新ResultSet

如果ResultSet是可更新的,则可以更新结果集中每一行的列。您可以使用许多updateXXX()方法来完成此操作。例如:

    result.updateString     ("name"       , "tianjh");result.updateInt        ("age"        , 26);result.updateBigDecimal ("money", new BigDecimal("100.0001");result.updateRow();

也可以使用列索引而不是列名来更新列。如下示例:

    result.updateString     (1, "tianjh");result.updateInt        (2, 26);result.updateBigDecimal (3, new BigDecimal("100.0001");result.updateRow();

请注意要使用updateRow() 调用。只有在调用updateRow() 时,数据库才会使用该行的值进行更新。如果您不调用此方法,则ResultSet中更新的值永远不会发送到数据库。如果在事务内部调用updateRow() ,那么在事务提交之前,数据实际上不会提交到数据库。

将行插入ResultSet中

如果ResultSet是可更新的,也可以在其中插入行

  1. 调用ResultSet.moveToInsertRow()
  2. 更新行-列值
  3. 调用ResultSet.insertRow()

以下是一个示例:

result.moveToInsertRow();
result.updateString     (1, "tianjh");
result.updateInt        (2, 26);
result.updateBigDecimal (3, new BigDecimal("100.0001");
result.insertRow();result.beforeFirst();

调用moveToInsertRow()后指向的行是一个特殊的行,一个缓冲区,您可以使用它来构建该行,直到该行上设置了所有列值。

一旦该行准备好插入ResultSet,就调用insertRow()方法。

插入行后,ResultSet仍指向插入行。但是,一旦插入该行,您无法确定如果尝试访问它会发生什么。因此,您应该在插入新行后将ResultSet移动到有效位置。如果需要插入另一行,请显式调用moveToInsertRow()以向ResultSet发出信号。

ResultSet可保持性

ResultSet的可保持性决定调用底层连接的commit()方法时是否关闭ResultSet

并非所有数据库和JDBC驱动程序都支持所有的可保持模式。DatabaseMetaData.supportsResultSetHoldability(int holdability)返回truefalse,具体取决于是否支持给定的holdability模式。DatabaseMetaData类将在后面的文本中介绍。

holdability有两种类型:

  1. ResultSet.CLOSE_CURSORS_OVER_COMMIT
  2. ResultSet.HOLD_CURSORS_OVER_COMMIT

CLOSE_CURSORS_OVER_COMMIT可保持性意味着当对创建ResultSet的连接调用connection.commit()方法时,所有ResultSet实例都将关闭。

HOLD_CURSORS_OVER_COMMIT可保持性意味着在创建ResultSet的连接上调用connection.commit()方法时,ResultSet保持打开状态。

如果使用ResultSet更新数据库中的值,那么HOLD_CURSORS_OVER_COMMIT保持能力可能会很有用。因此,您可以打开ResultSet,更新其中的行,调用connection.commit(),并且仍然为同一行上的未来事务打开相同的ResultSet

下一篇:JDBC PreparedStatement

这篇关于JDBC ResultSet的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1030963

相关文章

滚雪球学Java(87):Java事务处理:JDBC的ACID属性与实战技巧!真有两下子!

咦咦咦,各位小可爱,我是你们的好伙伴——bug菌,今天又来给大家普及Java SE啦,别躲起来啊,听我讲干货还不快点赞,赞多了我就有动力讲得更嗨啦!所以呀,养成先点赞后阅读的好习惯,别被干货淹没了哦~ 🏆本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,助你一臂之力,带你早日登顶🚀,欢迎大家关注&&收藏!持续更新中,up!up!up!! 环境说明:Windows 10

Hibernate框架中,使用JDBC语法

/*** 调用存储过程* * @param PRONAME* @return*/public CallableStatement citePro(final String PRONAME){Session session = getCurrentSession();CallableStatement pro = session.doReturningWork(new ReturningWork<C

jdbc连接数据库使用sid和service_name的区别 ?

问题描述: ORA-12505, TNS:listener does not currently know of SID given in connect descriptor The Connection descriptor used by the client was: 10.12.162.84:1521:xxxx  oracle数据的tnsnames.ora中配置的是:SERVICE

Java项目中,配置打印 JDBC 日志的几种方法

在 IDEA 项目中,如果你想打印 JDBC 日志,可以通过配置日志框架(如 Logback 或 Log4j)来实现。Spring Boot 使用的默认日志框架是 Logback,你可以通过在 application.yml 文件中配置日志级别来打印 JDBC 日志。 方法 1: 使用 application.yml 配置 JDBC 日志 logging:level:# 显示 SQL 语句co

基于shard-jdbc中间件,实现数据分库分表

一、水平分割 1、水平分库 1)、概念: 以字段为依据,按照一定策略,将一个库中的数据拆分到多个库中。 2)、结果 每个库的结构都一样;数据都不一样; 所有库的并集是全量数据; 2、水平分表 1)、概念 以字段为依据,按照一定策略,将一个表中的数据拆分到多个表中。 2)、结果 每个表的结构都一样;数据都不一样; 所有表的并集是全量数据; 二、Shard-jdbc 中间件 1、架构图 2、特点

基于Shard-Jdbc分库分表,数据库扩容方案

一、数据库扩容 1、业务场景 互联网项目中有很多“数据量大,业务复杂度高,需要分库分表”的业务场景。 这样分层的架构 (1)上层是业务层biz,实现业务逻辑封装; (2)中间是服务层service,封装数据访问; (3)下层是数据层db,存储业务数据; 2、扩容场景和问题 当数据量持续新增,面临着这样一些需求,两台数据库无法容纳,需要数据库扩容,这里选择2台—扩容到3台的模式,如下图

Java笔试面试题AI答之JDBC(3)

文章目录 13. 编写JDBC连Oracle的程序?14. 简述JDBC的主要组件有哪些 ?15. JDBC中如何防止SQL注入攻击?1. 使用预处理语句(PreparedStatement)2. 避免在SQL查询中直接拼接用户输入的数据总结 16. JDBC的脏读是什么?哪种数据库隔离级别能防止脏读?脏读(Dirty Read)哪种数据库隔离级别能防止脏读? 17. 简述JDBC ex

JavaBug系列- Failed to load driver class com.mysql.cj.jdbc.Driver in either of HikariConfig class load

JavaBug系列之Mysql驱动问题 Java医生一、关于错误信息二、如何解决问题 Java医生 本系列记录常见Bug,以及诊断过程和原因 Java/一对一零基础辅导/企业项目一对一辅导/日常Bug解决/代码讲解/毕业设计等 V:study_51ctofx 一、关于错误信息 APPLICATION FAILED TO START Description: Fai

Java笔试面试题AI答之JDBC(4)

文章目录 19. 解释JDBC的ResultSet是什么 ?20. JDBC编程有哪些不足之处,MyBatis是如何解决这些问题的?JDBC编程的不足之处MyBatis如何解决这些问题 21. 简述JDBC 能否处理 Blob 和 Clob ?1. JDBC对Blob和Clob的支持2. 处理Blob和Clob的示例3. 注意事项 19. 解释JDBC的ResultSet是什

Java学习记录(13)——JDBC

JDBC的全称是Java database connectivity,即Java数据库连接,它是一种可以执行SQL语句的Java API。 JDBC的典型用法 1、加载数据库驱动。通常使用Class类的forName()静态方法来加载驱动。 //加载mysql的驱动 Class.forName("com.mysql.jdbc.Driver"); //加载oracle的驱动 Cla