- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在开发一个 Java 项目,它在 Tomcat 6 上运行,连接到 MySQL 数据库。所有程序都按预期运行,无论是在本地测试还是在我们客户的服务器上进行测试。然而,有一个异常(exception),那就是一个过程,它检索大量数据以生成报告。从 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 文件(发生在更高级别)。
因此,如果有人对此有一些优化建议,我们非常欢迎。
最佳答案
好的,您的堆栈跟踪给了您答案:
Exception in thread "Thread-16" java.lang.OutOfMemoryError: Java heap space
这就是为什么你没有记录,应用程序正在崩溃(线程,具体来说)。从您的描述来看,您似乎有一个庞大的数据集需要分页。
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);
}
}
这就是线程死机的地方(可能)。基本上你的 StringBuffer 内存不足。至于纠正它,有很多选择。在客户端的问题上投入更多内存(通过配置 JVM(这是一个链接)): How to set the maximum memory usage for JVM?
或者,如果您已经在这样做,请将更多 RAM 放入设备中。
从编程的角度来看,这听起来像是一份糟糕的报告。您可以将一些数字处理卸载到 MySQL 而不是在您的终端进行缓冲(如果可能),或者,如果这是一个巨大的报告,我会考虑将其流式传输到一个文件,然后通过缓冲流读取以填充报告。
这完全取决于报告是什么。如果它很小,我的目标是在 SQL 中做更多的工作以最小化结果集。如果它是一个巨大的报告,那么缓冲是另一种选择。
您可能遗漏的另一种可能性是 ResultSet(取决于实现)可能已被缓冲。这意味着您的报告可以直接获取 ResultSet 对象并从中打印,而不是将其全部读取为字符串。当然,这样做的缺点是杂散的 SQL 异常会终止您的报告。
祝你好运,我会先尝试内存选项。您可能正在运行一些像 128 这样小得可笑的东西,而且它会很简单(我已经看到这种情况在远程管理的机器上经常发生)。
关于Java 存储过程因 OutOfMemoryError 而停止,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11690837/
我正在尝试将用户提供的经纬度值与数据库中的经纬度值进行比较。如果它们在彼此半径 15 公里内,则应更改 TextView 。但我面临以下错误, 我的数据库包含值 source lat = 19.218
我在我的应用程序中使用改造来下载一些媒体文件,如视频、mp3、jpg、pdf 等。当我想下载一个 55MB 的 mp4 格式的大文件时,这是一个问题。当我想下载这个文件时,我收到这样的错误: OutO
所以我正在创建一个 Android 应用程序,这段代码引发了 "Caused by: java.lang.OutOfMemoryError: OutOfMemoryError thrown while
直到昨天,我的应用程序运行良好,但我所做的是,由于某些原因,我不得不在 Android Studio 中打开具有不同工作空间的同一个应用程序。从那时起,当我尝试运行该应用程序时,我遇到了以下异常,所以
我正在尝试构建一个应用程序,其中客户端将其屏幕发送到服务器,客户端仅在上次发送屏幕和最新捕获的屏幕之间存在差异时才发送其屏幕(以便该程序在网络)。服务器使用 JFrame 和 JLabel 来显示图像
我正在尝试使用内存映射模式在 cupy 中加载一些较大的 .npy 文件,但我不断遇到 OutOfMemoryError 。 我认为,由于它是在内存映射模式下打开的,因此此操作不应该占用太多内存,因为
我正在尝试对基于 ant 的(Netbeans RCP)项目进行分级并找到奇怪的分级行为。 我用探查器做了一些观察,得到了下一个结果。 环境配置 Gradle 1.9 Build time: 20
我有一个应用程序可以进行网络调用并检索 XML 数据。如果没有太多数据,下面的代码可以正常工作。 public class WebClient { private static final S
在我的应用程序中,我每 3 分钟刷新一次数据。如果应用程序可以工作几个小时,我会遇到这样的错误: java.lang.OutOfMemoryError at org.apache.http.util.
我在我的一个应用程序中偶尔收到 OutOfMemoryError: (Heap Size=49187KB, Allocated=41957KB)。我该怎么做才能诊断? 01-09 10:32:02
对于学校项目,我必须编写不同类型的算法。问题是,我得到了一个工作算法。但是我必须多次运行它,一段时间后它给了我以下错误: Exception in thread "main" java.lang.Ou
这个问题在这里已经有了答案: 8年前关闭。 Possible Duplicate: Recursive function causing a stack overflow 完成示例惰性序列 here
我收到 java.lang.OutOfMemoryError 错误,即使我还有足够的空闲 RAM。我进行的内存转储在 200MB 到 1GB 之间,而我的服务器有 24GB 的 RAM。我设置了 -X
我不明白为什么这段代码没有OutOfMemoryError public static void main(String[] args) { Object[] ref = new Object
我正在使用这个语句 //some code int a[][]=new int[5000000][5000000]; //some code 并使用命令运行它 java -mx512m Test 它给
今天我在玩OOM错误,我发现了一些我自己无法解释的东西。 我尝试分配一个比堆大的数组,期望 “请求的阵列大小超出 VM 限制”错误,但我得到一个“ Java 堆空间 ”错误。 根据JDK 11 doc
我有一个显示图像的简单页面。来源是 URL var img = new Image (); var source = new UriImageSource { Uri =
我有一个 Java Spring Boot 应用程序。它是一个非常大的应用程序,具有许多服务,并且可以执行大量任务。我尝试实现的新任务之一是从 Oracle DB 读取一些数据并通过 REST 将其发
我正在尝试使用流读取一个非常大的文件,因此我需要并行流而不是每行迭代...我正在尝试如下: String cont = new String(Files.readAllBytes(Paths.get(
假设我们的最大内存为 256M,为什么这段代码可以工作: public static void main(String... args) { for (int i = 0; i < 2; i++)
我是一名优秀的程序员,十分优秀!