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