我正在开发一个 Java 项目,该项目在连接到 MySQL 数据库的 Tomcat 6 上运行。所有程序都按应有的方式运行,无论是在本地测试还是在我们客户的服务器上进行测试时。但是,有一个例外,即检索大量数据以生成报告的过程。从 MySQL 执行存储过程大约需要 13 分钟。当我在本地运行应用程序并连接到在线数据库时,该过程确实可以工作,唯一不工作的是它在我们客户端的服务器上运行时。
客户对他的服务器非常保护,所以我们对它的控制有限,但他们确实希望我们解决问题。当我检查日志文件时,执行存储过程的函数不会引发任何错误。并将一些调试日志放入代码中,它表明它确实到达了执行调用,但没有在调用后立即记录调试,也没有在 catch 中记录错误,但确实进入了 finally 部分。
他们声称 MySQL 日志中没有超时错误。
如果有人对可能导致此问题的原因有任何想法,我们将不胜感激。
更新:
在向服务器管理员唠叨了一番之后,我终于可以访问 catalina 日志了,在这些日志中,我终于发现了一个有意义的错误:
Exception in thread "Thread-16" java.lang.OutOfMemoryError: Java heap space
at java.util.Arrays.copyOf(Arrays.java:2894)
at java.lang.AbstractStringBuilder.expandCapacity(AbstractStringBuilder.java:117)
at java.lang.AbstractStringBuilder.append(AbstractStringBuilder.java:407)
at java.lang.StringBuffer.append(StringBuffer.java:241)
at be.playlane.mink.database.SelectExportDataProcedure.bufferField(SelectExportDataProcedure.java:68)
at be.playlane.mink.database.SelectExportDataProcedure.extractData(SelectExportDataProcedure.java:54)
at org.springframework.jdbc.core.JdbcTemplate.processResultSet(JdbcTemplate.java:1033)
at org.springframework.jdbc.core.JdbcTemplate.extractReturnedResultSets(JdbcTemplate.java:947)
at org.springframework.jdbc.core.JdbcTemplate$5.doInCallableStatement(JdbcTemplate.java:918)
at org.springframework.jdbc.core.JdbcTemplate.execute(JdbcTemplate.java:876)
at org.springframework.jdbc.core.JdbcTemplate.call(JdbcTemplate.java:908)
at org.springframework.jdbc.object.StoredProcedure.execute(StoredProcedure.java:113)
at be.playlane.mink.database.SelectExportDataProcedure.execute(SelectExportDataProcedure.java:29)
at be.playlane.mink.service.impl.DefaultExportService$ExportDataRunnable.run(DefaultExportService.java:82)
at java.lang.Thread.run(Thread.java:636)
奇怪的是这不会记录到应用程序日志中,即使它被包装在一个 try catch 中。现在基于错误,问题出在这种方法上:
public Object extractData(ResultSet rs) throws SQLException, DataAccessException
{
StringBuffer buffer = new StringBuffer();
try
{
// get result set meta data
ResultSetMetaData meta = rs.getMetaData();
int count = meta.getColumnCount();
// get the column names; column indices start from 1
for (int i = 1; i < count + 1; ++i)
{
String name = meta.getColumnName(i);
bufferField(name, i == count, buffer);
}
while (rs.next())
{
// get the column values; column indices start from 1
for (int i = 1; i < count + 1; ++i)
{
String value = rs.getString(i);
bufferField(value, i == count, buffer);
}
}
}
catch (Exception e)
{
logger.error("Failed to extractData SelectExportDataProcedue: ", e);
}
return buffer.toString();
}
private void bufferField(String field, boolean last, StringBuffer buffer)
{
try
{
if (field != null)
{
field = field.replace('\r', ' ');
field = field.replace('\n', ' ');
buffer.append(field);
}
if (last)
{
buffer.append('\n');
}
else
{
buffer.append('\t');
}
}
catch (Exception e)
{
logger.error("Failed to bufferField SelectExportDataProcedue: ", e);
}
}
这些函数的目标是将某个结果集导出到 excel 文件(这发生在更高级别)。
因此,如果有人对此有一些优化建议,我们将非常欢迎。