JDBC-ResultSet 在 while-Loop 中关闭

Posted

技术标签:

【中文标题】JDBC-ResultSet 在 while-Loop 中关闭【英文标题】:JDBC-ResultSet is closed in while-Loop 【发布时间】:2014-09-17 19:32:40 【问题描述】:

我在使用 ResultSet 时遇到了非常糟糕的情况,该 ResultSet 在用于迭代此 ResultSet 的 while 循环内关闭。我知道 ResultSet 关闭的确切行,但我不知道为什么。

    public LinkedList<Athlet> alleAbrufen () throws SQLException 
        LinkedList<Athlet> alleAthleten = new LinkedList<Athlet>();
        String abrufenAthleten = "SELECT * FROM Athlet ORDER BY athlet_id";
        ResultSet athleten_rs = stmt.executeQuery(abrufenAthleten);
        while (athleten_rs.next()) 
            long id = athleten_rs.getInt(1);
            String name = athleten_rs.getString(2);
            LinkedList<Leistung> alleLeistungen = alleAbrufen((int) (id)); //after this line the ResultSet gets closed
            alleAthleten.add(new Athlet(id, name, alleLeistungen));
        
        return alleAthleten;
    

    public LinkedList<Leistung> alleAbrufen(int athlet_id) throws SQLException 
        LinkedList<Leistung> alleLeistungen = new LinkedList<Leistung>();
        String selectLeistungen = "SELECT * FROM Leistung WHERE athlet_id="+athlet_id;
        ResultSet rs = stmt.executeQuery(selectLeistungen);
        while (rs.next()) 
            long id = rs.getInt(1); 
            String bezeichnung = rs.getString(2);
            String datum = rs.getString(3);
            double geschwindigkeit = rs.getDouble(4);
            boolean selectedForSlopeFaktor = rs.getBoolean(5);
            int strecke_id = rs.getInt(7);
            long longAthlet_id = (long) athlet_id;
            Leistung leistung = new Leistung(strecke_id, longAthlet_id, bezeichnung, datum, geschwindigkeit);
            leistung.setLeistungID(id);
            leistung.setIsUsedForSlopeFaktor(selectedForSlopeFaktor);
            alleLeistungen.add(leistung);
        
        return alleLeistungen;
    

我标记了ResultSet 以注释关闭之后的行。 Alle 上面示例中使用的其他方法、构造函数等经过测试可以正常工作。有谁知道为什么调用第二种方法会关闭第一种方法中的ResultSet

【问题讨论】:

【参考方案1】:

问题是Statement 在每个执行的语句中只能维护一组ResultSets。由于您为两种方法共享相同的Statement stmt,因此在alleAbrufenStatement 执行另一个语句,这将破坏对先前ResultSet 的引用。

这种情况的最佳解决方案是为每个语句执行创建一个Statement。也就是说,每个方法都应该包含其唯一的Statement 和相关的ResultSets。

public LinkedList<Athlet> alleAbrufen () throws SQLException 
    LinkedList<Athlet> alleAthleten = new LinkedList<Athlet>();
    String abrufenAthleten = "SELECT * FROM Athlet ORDER BY athlet_id";
    //here
    Statement stmtAlleAbrufen = con.createStatement();
    ResultSet athleten_rs = stmtAlleAbrufen.executeQuery(abrufenAthleten);
    while (athleten_rs.next()) 
        long id = athleten_rs.getInt(1);
        String name = athleten_rs.getString(2);
        LinkedList<Leistung> alleLeistungen = alleAbrufen((int) (id)); //after this line the ResultSet gets closed
        alleAthleten.add(new Athlet(id, name, alleLeistungen));
    
    return alleAthleten;


public LinkedList<Leistung> alleAbrufen(int athlet_id) throws SQLException 
    LinkedList<Leistung> alleLeistungen = new LinkedList<Leistung>();
    //here again, but since you need to use parameters in your query
    //use PreparedStatement instead
    //note that I commented the current query
    //String selectLeistungen = "SELECT * FROM Leistung WHERE athlet_id="+athlet_id;
    //this is how a query with parameters look like
    String selectLeistungen = "SELECT * FROM Leistung WHERE athlet_id=?";
    //the connection prepares the statement
    PreparedStatement pstmt = con.prepareStatement(selectLeistungen);
    //then we pass the parameters
    pstmt.setInt(1, athlet_id);
    ResultSet rs = pstmt.executeQuery();
    while (rs.next()) 
        long id = rs.getInt(1); 
        String bezeichnung = rs.getString(2);
        String datum = rs.getString(3);
        double geschwindigkeit = rs.getDouble(4);
        boolean selectedForSlopeFaktor = rs.getBoolean(5);
        int strecke_id = rs.getInt(7);
        long longAthlet_id = (long) athlet_id;
        Leistung leistung = new Leistung(strecke_id, longAthlet_id, bezeichnung, datum, geschwindigkeit);
        leistung.setLeistungID(id);
        leistung.setIsUsedForSlopeFaktor(selectedForSlopeFaktor);
        alleLeistungen.add(leistung);
    
    return alleLeistungen;

使用资源后不要忘记关闭StatementResultSet

【讨论】:

感谢您的回答!它对我有用。两次使用相同的 ResultSet 是一种愚蠢的错误……无论如何,谢谢,Luiggi! @Shikari 你使用了同一个 Statement 对象两次。【参考方案2】:

您的问题的答案来自 javadoc:

默认情况下,每个 Statement 对象只能打开一个 ResultSet 对象 同时。因此,如果读取一个 ResultSet 对象是 与另一个阅读交错,每个必须已经生成 通过不同的 Statement 对象。

【讨论】:

【参考方案3】:

您的Statement 是一个类变量吗,并且您对两个查询都使用相同的变量?是的,这是错误的。每个Statement 只能有一个ResultSet

请参阅java docs。

【讨论】:

以上是关于JDBC-ResultSet 在 while-Loop 中关闭的主要内容,如果未能解决你的问题,请参考以下文章

分配的变量引用在哪里,在堆栈中还是在堆中?

NOIP 2015 & SDOI 2016 Round1 & CTSC 2016 & SDOI2016 Round2游记

秋的潇洒在啥?在啥在啥?

上传的数据在云端的怎么查看,保存在啥位置?

在 React 应用程序中在哪里转换数据 - 在 Express 中还是在前端使用 React?

存储在 plist 中的数据在模拟器中有效,但在设备中无效