马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。
您需要 登录 才可以下载或查看,没有账号?立即注册
×
问题描述
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(游标类型,记录更新权限);
解决办法
<ol>手动指定游标类型
Statement stmt =conn.createStatement(ResultSet.TYPE_SCOLL_INSENSITIVE,ResultSet.CONCUR_READ_ONLY);
将 ResultSet 的内容复制到一个 List 中,然后检查 List 是否为空
[code]List results = new ArrayList();while (rs.next()) { int columnCount = rs.getMetaData().getColumnCount(); Object[] row = new Object[columnCount]; for (int i = 1; i
|