gpt4 book ai didi

java - 结果集存储在 hashMap 中,行数为零

转载 作者:行者123 更新时间:2023-12-01 11:24:04 26 4
gpt4 key购买 nike

我的类中有两个方法,首先我调用方法 dbExecuteStatement() ,执行sql查询。执行sql查询后,我得到一个ResultSet对象。我将此 ResultSet 对象保存在静态 hashMap 中,以便在我的下一个方法调用 fetchResults() 中,我可以使用现有的结果集来检索结果。将 ResultSet 对象保存在映射中的原因是 fetchResults()方法请求参数,我将获取最大获取行大小,并根据该值迭代结果集。这两种方法都应该从客户端单独调用。

现在我面临的问题是,当我迭代 fetchResults() 中的 ResultSet 对象时方法,我得到的行数为零。如果我从 dbExecuteStatement() 中的 hashMap 获取相同的结果集,我得到实际的行数,在我的例子中即 5 。我检查了放入 fetchResults() 中的 HashMap 的 ResultSet 对象方法和dbExecuteStatement() ,它是同一个对象。但如果得到 ResultSetMetaData fetchResults() 中的对象方法和dbExecuteStatement() ,他们来的方式不同。有人可以帮助我理解原因,为什么我的结果计数为零。

下面是代码:

  public class HiveDao1 {
private static Map<Object,Map<Object,Object>> databaseConnectionDetails
= new HashMap<Object,Map<Object,Object>>();

//This method will execute the sql query and will save the ResultSet obj in a hashmap for later use
public void dbExecuteStatement(DbExecuteStatementReq dbExecuteStatementReq){
//I already have a connection object saved in map
String uniqueIdForConnectionObject = dbExecuteStatementReq.getDbUniqueConnectionHandlerId();
Map<Object,Object> dbObject = databaseConnectionDetails.get(uniqueIdForConnectionObject);
Connection connection = (Connection) dbObject.get(DatabaseConstants.CONNECTION);

try {

Statement stmt = connection.createStatement() ;
// Execute the query
ResultSet resultSet = stmt.executeQuery(dbExecuteStatementReq.getStatement().trim()) ;

//save the result set for further use, Result set will be used in fetchResult() call
dbObject.put(DatabaseConstants.RESULTSET, resultSet);

/*
* Now below is the debugging code,which I put to compare the result set
* iteration dbExecuteStatement() and fetchResults method
*/
ResultSet rs = (ResultSet) dbObject.get(DatabaseConstants.RESULTSET);

ResultSetMetaData md = (ResultSetMetaData) dbObject.get(DatabaseConstants.RESULTSETMETADATA);

System.out.println("==ResultSet fethced in dbExecuteStatement=="+rs);
System.out.println("==ResultSet metadata fetched in dbExecuteStatement ==="+rs.getMetaData());

int count = 0;
while (rs.next()) {
++count;
}

if (count == 0) {
System.out.println("No records found");
}

System.out.println("No of rows found from result set in dbExecuteStatement is "+count);

} catch (SQLException e) {
e.printStackTrace();
}

}

/*
* This method fetch the result set object from hashMap
* and iterate it on the basis of fetch size received in req parameter
*/
public void fetchResults(FetchResultsReq fetchResultsReq){

String uniqueIdForConnectionObject = fetchResultsReq.getDbUniqueConnectionHandlerId();
Map<Object,Object> dbObject = databaseConnectionDetails.get(uniqueIdForConnectionObject);

try {
//Fetch the ResultSet object that was saved by dbExecuteStatement()
ResultSet rs = (ResultSet) dbObject.get(DatabaseConstants.RESULTSET);
ResultSetMetaData md = (ResultSetMetaData) dbObject.get(DatabaseConstants.RESULTSETMETADATA);

System.out.println("ResultSet fethced in fetchResults at server side dao layer======"+rs);
System.out.println("ResultSet metadata fetched in fetchResults at server side dao layer======"+md);

int count = 0;
while (rs.next()) {
++count;
}

if (count == 0) {
System.out.println("No records found");
}
//Here the row count is not same as row count in dbExecuteStatement()
System.out.println("No of rows found from result set in fetchResults is "+count);

} catch (SQLException e) {
e.printStackTrace();
}
}

}

最佳答案

扩展我的评论(和@Glenn 的):

多次使用 ResultSet

当您编写迭代 ResultSet 的调试代码时,光标会移动到结果的末尾。当然,如果您随后调用同一个对象并使用 next(),它仍然会位于末尾,因此您不会再获得任何记录。

如果您确实需要多次读取同一个 ResultSet,则需要执行查询以返回可滚动的 ResultSet。您在创建语句时执行此操作:

Statement stmt = connection.createStatement( 
ResultSet.TYPE_SCROLL_INSENSITIVE,
ResultSet.CONCUR_READ_ONLY );

不带参数的 connection.createStatement() 创建的默认语句返回 ResultSet.TYPE_FORWARD_ONLY 类型的结果集,以及 ResultSet 对象只能读取一次。

如果你的结果集类型是滚动不敏感或滚动敏感,你可以使用像rs.first()这样的语句来重置游标,然后你就可以再次获取记录。

将声明保持在范围内

@Glenn 的评论非常重要。按照您的程序现在的工作方式,它可能在整个测试阶段都工作正常,然后突然在生产中,您的 ResultSet 中有时会出现零记录,并且错误只会偶尔重现 -调试噩梦。

如果生成 ResultSetStatement 对象关闭,则 ResultSet 本身也会关闭。由于您没有自己关闭 Statement 对象,因此这将在 Statement 对象最终确定时完成。

stmt 变量是本地变量,并且它是我们所知的对该 Statement 的唯一引用。因此,它将被垃圾收集器认领。然而,具有终结器的对象被归入终结队列,并且无法知道终结器何时被调用,也无法控制它。一旦发生这种情况,ResultSet 将被关闭,不受您的控制。

因此请务必在 ResultSet 旁边保留对语句对象的引用。确保在使用完 ResultSet 并且不再使用它后自行正确关闭它。关闭它后,请记住删除保留的引用 - 无论是语句还是结果集 - 以避免内存泄漏。关闭很重要,依赖终结器是一个糟糕的策略。如果您不自己关闭它,您可能会在数据库中的某个时刻用完游标(取决于 DBMS 及其配置)。

关于java - 结果集存储在 hashMap 中,行数为零,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30965016/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com