详解Java如何判断ResultSet结果集是否为空

 更新时间:2023年02月10日 08:43:38   作者:奥怪的小栈  
ResultSet 表示 select 语句的查询结果集。这篇文章主要为大家详细介绍了Java如何判断ResultSet结果集是否为空,感兴趣的可以了解一下

问题描述

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(...);

到此这篇关于详解Java如何判断ResultSet结果集是否为空的文章就介绍到这了,更多相关Java判断ResultSet是否为空内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java如何将BigDecimal类型的值转成double类型

    Java如何将BigDecimal类型的值转成double类型

    这篇文章主要给大家介绍了关于Java如何将BigDecimal类型的值转成double类型的相关资料,需要注意精度损失和范围限制,使用doubleValue方法进行转换,并在高精度计算时格外小心,需要的朋友可以参考下
    2024-12-12
  • Java中实现多线程关键词整理(总结)

    Java中实现多线程关键词整理(总结)

    这篇文章主要介绍了Java中实现多线程关键词整理,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2017-05-05
  • Redis在springboot中的使用教程

    Redis在springboot中的使用教程

    这篇文章主要介绍了Redis在springboot中的使用教程,本文实例代码相结合的形式给大家介绍的非常详细,需要的朋友可以参考下
    2018-06-06
  • 详解HDFS多文件Join操作的实例

    详解HDFS多文件Join操作的实例

    这篇文章主要介绍了详解HDFS多文件Join操作的实例的相关资料,希望通过本文能帮助到大家,让大家理解掌握这部分内容,需要的朋友可以参考下
    2017-10-10
  • 最详细的文件上传下载实例详解(推荐)

    最详细的文件上传下载实例详解(推荐)

    在Web应用系统开发中,文件上传和下载功能是非常常用的功能,今天来讲一下JavaWeb中的文件上传和下载功能的实现。非常不错,具有参考借鉴价值,感兴趣的朋友一起看下吧
    2016-07-07
  • java学生管理系统界面简单实现(全)

    java学生管理系统界面简单实现(全)

    这篇文章主要为大家详细介绍了java学生管理系统界面的简单实现,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-01-01
  • SpringMVC实现获取请求参数方法详解

    SpringMVC实现获取请求参数方法详解

    Spring MVC 是 Spring 提供的一个基于 MVC 设计模式的轻量级 Web 开发框架,本质上相当于 Servlet,Spring MVC 角色划分清晰,分工明细,这篇文章主要介绍了SpringMVC实现获取请求参数方法
    2022-09-09
  • SpringBoot多环境配置及日志记录器详解

    SpringBoot多环境配置及日志记录器详解

    这篇文章主要介绍了SpringBoot多环境配置及日志记录器详解,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧
    2024-01-01
  • 详解Nacos配置中心的实现

    详解Nacos配置中心的实现

    Spring Cloud Alibaba 是阿里巴巴提供的一站式微服务开发解决方案。而 Nacos 作为 Spring Cloud Alibaba 的核心组件之一,提供了两个非常重要的功能:注册中心和配置中心,我们今天来了解和实现一下二者
    2022-08-08
  • Java线程池ThreadPoolExecutor源码深入分析

    Java线程池ThreadPoolExecutor源码深入分析

    ThreadPoolExecutor作为java.util.concurrent包对外提供基础实现,以内部线程池的形式对外提供管理任务执行,线程调度,线程池管理等等服务
    2022-08-08

最新评论