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(...);
    
  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
回答: 要获取java.sql.ResultSet的结果行数,可以使用以下两种方法。第一种方法是使用ResultSet的next()方法来遍历结果,直到光标移动到最后一行,然后使用ResultSet的getRow()方法获取当前行的行号。如果结果为空,getRow()方法将返回0。\[1\]第二种方法是使用ResultSet的last()方法将光标移动到最后一行,然后使用getRow()方法获取当前行的行号。如果结果为空,getRow()方法将返回0。\[3\]请注意,如果结果的类型是TYPE_FORWARD_ONLY,那么在调用next()方法返回false后,光标将位于最后一行之后,此时再调用任何需要当前行的ResultSet方法将抛出SQLException异常。\[2\] #### 引用[.reference_title] - *1* *2* [java.sql.ResultSet.next() 光标常见错误总结分析](https://blog.csdn.net/weixin_34855166/article/details/115073437)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [How do I get the size of a java.sql.ResultSet?](https://blog.csdn.net/qijingpei/article/details/116381266)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值