Warning: file_get_contents(/data/phpspider/zhask/data//catemap/1/oracle/10.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
java.sql.SQLException关闭语句_Java_Oracle_Jdbc_Transactions_Sqlexception - Fatal编程技术网

java.sql.SQLException关闭语句

java.sql.SQLException关闭语句,java,oracle,jdbc,transactions,sqlexception,Java,Oracle,Jdbc,Transactions,Sqlexception,我有一个问题,如果我在没有autoCommit属性的情况下运行,下面的代码运行得很好,但是我更喜欢将其作为事务运行,代码基本上会插入一篇文章的标题信息,然后插入与之关联的每个文章的列表(因此它就像一对多的关系),因此,我可以一次性提交所有内容,而不是先提交文章信息,然后提交项目。问题是,当我到达cn.commit()行时,我得到一个异常,上面写着“Closed Statement” 数据库插入方法 public static void addArticle(Article article) th

我有一个问题,如果我在没有autoCommit属性的情况下运行,下面的代码运行得很好,但是我更喜欢将其作为事务运行,代码基本上会插入一篇文章的标题信息,然后插入与之关联的每个文章的列表(因此它就像一对多的关系),因此,我可以一次性提交所有内容,而不是先提交文章信息,然后提交项目。问题是,当我到达cn.commit()行时,我得到一个异常,上面写着“Closed Statement”

数据库插入方法

public static void addArticle(Article article) throws SQLException {
             Connection cn = null;
             PreparedStatement ps = null;
             StringBuffer insert = new StringBuffer();
             StringBuffer itemsSQL = new StringBuffer();

             try {
                    article.setArticleSortNum(getNextArticleNum(article.getShopId()));
                    article.setArticleId(DAOHelper.getNextId("article_id_sequence"));

                    cn = DBHelper.makeConnection();
                    cn.setAutoCommit(false);

                    insert.append("insert query for article goes here");
                    ps = cn.prepareStatement(insert.toString());
                    int i = 1;
                    ps.setLong(i, article.getArticleId()); i++;
                    ps.setLong(i, article.getShopId()); i++;
                    ps.setInt(i, article.getArticleNum()); i++;
                    // etcetera...
                    ps.executeUpdate(); 

                    itemsSQL.append("insert query for each line goes here");
                    itemStatement = cn.prepareStatement(itemsSQL.toString());
                    for(Article item : article.getArticlesList()) {

                        item.setArticleId(article.getArticleId());
                        i= 1;
                        itemStatement.setLong(i, item.getArticleId()); i++;
                        itemStatement.setInt(i,  item.getItemsOnStock()); i++;
                        itemStatement.setInt(i,  item.getQuantity()); i++;
                        // etcetera...
                        itemStatement.executeUpdate();

                    }
                    cn.commit();

             } catch (SQLException e) {
                    cn.rollback();
                    log.error(e.getMessage());
                    throw e;
             }
             finally {
                    DBHelper.releasePreparedStatement(ps);
                    DBHelper.releasePreparedStatement(itemStatement);
                    DBHelper.releaseConnection(cn);
             }
       }
我还使用addBatch()和executeBatch插入了For运行的项,但在到达cn.commit()时也出现了相同的Closed语句错误。。。我不明白为什么它关闭了,所有的连接和所有的东西都在finally子句中被释放了,所以我感觉我犯了一些根本性的错误,我没有意识到。。。有什么想法吗?提前谢谢

编辑:下面是堆栈跟踪:

java.sql.SQLException:Closed语句位于 位于的oracle.jdbc.dbaccess.DBError.throwSqlException(DBError.java:189) 位于的oracle.jdbc.dbaccess.DBError.throwSqlException(DBError.java:231) 位于的oracle.jdbc.dbaccess.DBError.throwSqlException(DBError.java:294) oracle.jdbc.driver.OracleStatement.ensureOpen(OracleStatement.java:6226) 在 oracle.jdbc.driver.OraclePreparedStatement.sendBatch(OraclePreparedStatement.java:592) 在 oracle.jdbc.driver.OracleConnection.commit(OracleConnection.java:1376) 在com.evermind.sql.FilterConnection.commit(FilterConnection.java:201)上 在 com.evermind.sql.OrionCMTConnection.commit(OrionCMTConnection.java:461) 在com.evermind.sql.FilterConnection.commit(FilterConnection.java:201)上 在com.dao.ArticlesDAO.addArticle(ArticlesDAO.java:571)上 com.action.registry.CustomBaseAction.execute(CustomBaseAction.java:57) 在 org.apache.struts.action.RequestProcessor.processActionPerform(RequestProcessor.java:431) 在 org.apache.struts.action.RequestProcessor.process(RequestProcessor.java:236) 在 org.apache.struts.action.ActionServlet.process(ActionServlet.java:1196) 在 org.apache.struts.action.ActionServlet.doPost(ActionServlet.java:432) 位于javax.servlet.http.HttpServlet.service(HttpServlet.java:760) http.HttpServlet.service(HttpServlet.java:853)位于 com.evermind.server.http.ServletRequestDispatcher.invoke(ServletRequestDispatcher.java:765) 在 com.evermind.server.http.ServletRequestDispatcher.forwardInternal(ServletRequestDispatcher.java:317) 在 com.evermind.server.http.HttpRequestHandler.processRequest(HttpRequestHandler.java:790) 在 com.evermind.server.http.HttpRequestHandler.run(HttpRequestHandler.java:270) 在 com.evermind.server.http.HttpRequestHandler.run(HttpRequestHandler.java:112) 在 com.evermind.util.ReleasableResourcePooledExecutor$MyWorker.run(ReleasableResourcePooledExecutor.java:192) 位于java.lang.Thread.run(未知源)

编辑2:

这些是驱动程序的datasource配置中的参数,我认为调试过程可能会超时,但即使在不到一秒钟内完成,也会抛出closed语句异常


通常最好是创建一个语句,使用它并尽快关闭它,在事务提交之前这样做并没有坏处。从阅读有关批处理模型的Oracle教程来看,一次打开多个语句似乎是个问题。在使用itemStatement之前,我会尝试关闭ps对象,然后移动初始化

itemStatement = cn.prepareStatement(itemsSQL.toString());
要在for循环的正上方移动,并将itemStatement移动到for循环之后立即关闭的位置:

PreparedStatement itemStatement = cn.prepareStatement(itemsSQL.toString());
try {
    for(Article item : article.getArticlesList()) {
       item.setArticleId(article.getArticleId());
        i= 1;
        itemStatement.setLong(i, item.getArticleId()); i++;
        itemStatement.setInt(i,  item.getItemsOnStock()); i++;
        itemStatement.setInt(i,  item.getQuantity()); i++;
                    // etcetera...
        itemStatement.executeUpdate();
    }
} finally {
    DBHelper.releasePreparedStatement(itemStatement);
}
看起来发生的事情是,在连接上设置了一些批处理参数,导致连接尝试在语句中查找未完成的事务以完成;它发现该语句已关闭,连接正在对此进行抱怨。这很奇怪,因为在commit爆炸的时候,代码还没有到达语句关闭的最终位置


阅读Oracle批处理模型可能会有所帮助。另外,请检查JDBC驱动程序版本,确保它适用于您正在使用的Oracle版本,并查看是否有任何可用的更新。

您可以发布异常的完整堆栈跟踪吗?我已经编辑了问题并添加了堆栈跟踪,看起来确实可以使用!但我得到了完全相同的例外:(我已经更新了配置文件中的超时/连接参数has@Ricardo:好的。请查看我的更新是否有用。此外,如果您在代码或连接配置中有关于批处理的更多详细信息,请将其添加到问题中。在for循环中创建语句将消除准备一次的优点,而不是引入“每次迭代的开销是多少。@马克:谢谢,我会更新。关于批处理部分有什么想法吗?在itemstatement打开时保持ps语句打开是否可能会导致问题?关于这个问题的信息是否不够?”?
PreparedStatement itemStatement = cn.prepareStatement(itemsSQL.toString());
try {
    for(Article item : article.getArticlesList()) {
       item.setArticleId(article.getArticleId());
        i= 1;
        itemStatement.setLong(i, item.getArticleId()); i++;
        itemStatement.setInt(i,  item.getItemsOnStock()); i++;
        itemStatement.setInt(i,  item.getQuantity()); i++;
                    // etcetera...
        itemStatement.executeUpdate();
    }
} finally {
    DBHelper.releasePreparedStatement(itemStatement);
}