乐趣区

关于java: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(...);
退出移动版