详解Java如何判断ResultSet结果集是否为空
作者:奥怪的小栈 发布时间:2023-08-22 10:51:14
问题描述
ResultSet 表示 select 语句的查询结果集。ResultSet 对象具有指向其当前数据行的指针, 最初,指针被置于第一行记录之前,通过 next() 方法可以将指针移动到下一行记录。
next() 方法在 ResultSet 对象没有一行记录时返回 false ,因此可以在 while 循环中使用它来遍历结果集,也可以利用该方法判断结果集是否为空。
示例代码如下:
//此处省略连接数据库的代码...
Statement stmt =conn.createStatement();
ResultSet rs =stmr.executeQuery("select * from Test");
if(rs.next()){
System.out.println("结果集不为空!");
}
else{
System.out.println("结果集为空!");
}
此时出现第一个坑:Java 的 ResultSet 对象,默认是不可更新的,仅有一个向前移动的指针。
因此,只能遍历它一次,并且只能按从第一行到最后一行的顺序进行。
当你使用了 rs.next() 进行判断后,会出现第一行数据丢失的情况。
这也是我一开始遇到的问题。
深究问题
ResultSet 的 Type 属性
遇到问题后我第一想法是在搜索引擎上搜索相关解决办法,但看了一圈具体有以下“解法”:
调用 rs.last() 方法,以获取 ResultSet中 记录的总数,然后调用 rs.beforeFirst() 方法将光标移回到第一条记录前面
这种方法看上去可行,但当我实际修改后运行,却出现报错
Operation not allowed for a result set of type ResultSet.TYPE_FORWARD_ONLY.
这又是什么原因呢?
出现这个报错的主要原因是:
ResultSet.TYPE_FORWARD_ONLY 类型的 ResultSet 只允许向前遍历,不支持访问先前的记录或确定其大小。因此,使用 last() 和 getRow() 等方法都是不可行。
而 ResultSet 的 Type 属性有如下几种:
参数类型 | 说明 |
---|---|
ResultSet.TYPE_FORWORD_ONLY | 结果集的游标只能向下滚动 |
ResultSet.TYPE_SCROLL_INSENSITIVE | 结果集的游标可以上下移动,当数据库变化时,当前结果集不变 |
ResultSet.TYPE_SCROLL_SENSITIVE | 返回可滚动的结果集,当数据库变化时,当前结果集同步改变 |
当 Statement stmt = conn.createStatement();
中 的 createStatement() 缺省时等价于: createStatement(ResultSet.TYPE_FORWORD_ONLY,ResultSet.CONCUR_READ_ONLY);
也就是结果集的游标只能向下滚动
所以才会出现 Operation not allowed for a result set of type ResultSet.TYPE_FORWARD_ONLY. 的报错
也就是 ResultSet 默认情况下,只能使用 next() 方法向前逐行移动游标,而不支持 last()、first() 以及 absolute() 等方法,如果要使用 last()、absolute() 等方法,必须在由 Connection 生成 Statement 时指定相应的参数,格式如下:
Statement stmt =conn.ctrateStatement(游标类型,记录更新权限);
解决办法
1.手动指定游标类型
Statement stmt =conn.createStatement(ResultSet.TYPE_SCOLL_INSENSITIVE,ResultSet.CONCUR_READ_ONLY);
2.将 ResultSet 的内容复制到一个 List 中,然后检查 List 是否为空
List<Object[]> results = new ArrayList<>();
while (rs.next()) {
int columnCount = rs.getMetaData().getColumnCount();
Object[] row = new Object[columnCount];
for (int i = 1; i <= columnCount; i++) {
row[i - 1] = rs.getObject(i);
}
results.add(row);
}
if (!results.isEmpty()) {
// ResultSet不为空,可以执行读取操作
} else {
// ResultSet为空,不执行读取操作
}
请注意,在实际生产环境中,请考虑数据量和内存使用情况,因为将所有记录复制到List中可能对内存产生很大影响。
3.定义 一个计数变量 i ,每次 next() 则 ++i ,在 while() 循环结束后判断 i 是否小于等于 0
int i = 0;
while (resultSet.next()){
System.out.println(...);
++i;
}
if(i<=0)
System.out.println(...);
来源:https://segmentfault.com/a/1190000043407293


猜你喜欢
- 问题(1)synchronized的特性?(2)synchronized的实现原理?(3)synchronized是否可重入?(4)sync
- 1.概述在本快速教程中,我们将演示如何在Spring Boot应用程序中自定义Spring Security的身份验证失败处理。目标是使用表
- 发现问题最近在进行压测发现,有一些接口时好时坏,通过sentry日志平台及sky walking平台跟踪发现,用户张三获取到的用户上下文确是
- 前言此前部门内的一个线上系统上线后内存一路飙高、一段时间后直接占满。协助开发人员去分析定位,发现内存中某个Object的量远远超出了预期的范
- 一、前言canal:阿里巴巴 MySQL binlog 增量订阅&消费组件https://github.com/alibaba/ca
- 前言大部分来自:https://blog.csdn.net/justloveyou_/article/details/61672133。并在
- 最近有小伙伴告诉我,在循环的判断条件只会计算一次,本金鱼不相信,于是就做了测试,本文记录我做的测试。先来写一个简单的代码, 就一个循环,循环
- 废话不多说,直接给大家贴C#代码了。/// <summary>/// 执行存储过程,返回" 返回值"///
- Android手势解锁本文讲述的是一个手势解锁的库,可以定制显示隐藏宫格点、路径、并且带有小九宫格显示图,和震动!让你学会使用这个简单,高效
- C# 获取硬件参数的实现方法示例代码:private static string GetIdentifier(string wmiClass
- String和List<String>间相互转换public void test() {  
- AIDL是Android接口定义语言,它可以用于让某个Service与多个应用程序组件之间进行跨进程通信,从而可以实现多个应用程序共享同一个
- 记录Java执行groovy脚本的两种方式,简单粗暴:一种是通过脚本引擎ScriptEngine提供的eval(String)方法执行脚本内
- 注:图片来源于网络SpringBoot作为业内公认的优秀开源框架,它的 * 是如何实现呢?在这里首先对一些基础组件进行分析;1、事件Appl
- 实现闹钟运行的效果如下: 通知栏的运行后效果图如下:布局文件(activity_main.xml)<?xml version
- 问题介绍:用二维数组表示一个迷宫,设置迷宫起点和终点,输出迷宫中的一条通路实现思路:二维数组表示迷宫:0表示路且未走过、1表示墙、2表示通路
- 要让项目实现 ssl 免密登录,首先需要开启 https 。所以先从 Spring Boot 如何开启 https 说起。创建服务端证书为了
- 前言一个简单的单机小游戏:flypybird ,用来巩固java基础。涉及主要知识点:JFrame 、 JPanel 、 继承、 键盘/鼠标
- 符号 ASCII码 &
- 微信支付现在已经变得越来越流行了,随之也出现了很多以可以快速接入微信支付为噱头的产品,不过方便之余也使得我们做东西慢慢依赖第三方,丧失了独立