- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我最近在 Java 中创建了一个方法来获取字符串的排列,但是当字符串太长时它会抛出这个错误:java.lang.OutOfMemoryError: Java heap space我确信该方法是有效的,因此我需要有关如何分散计算以避免错误的建议。我在 Eclipse 中使用控制台运行它。
public static ArrayList<String> permutation(String s) {
ArrayList<String> res = new ArrayList<String>();
if (s.length() == 1) {
res.add(s);
} else if (s.length() > 1) {
int lastIndex = s.length() - 1;
String last = s.substring(lastIndex);
String rest = s.substring(0, lastIndex);
res = merge(permutation(rest), last);
}
return res;
}
public static int factorial(int n) {
int fact = 1;
for (int i = 1; i <= n; i++) {
fact *= i;
}
return fact;
}
public static ArrayList<String> merge(ArrayList<String> list, String c) {
ArrayList<String> res = new ArrayList<String>();
for (String s : list) {
for (int i = 0; i <= s.length(); ++i) {
String ps = new StringBuffer(s).insert(i, c).toString();
res.add(ps);
}
}
return res;
}
最佳答案
首先,您需要确保了解导致 OutOfMemoryError (OOME) 的原因。您多次引用了随着时间的推移展开计算。我可能会误解你的意思,但如前所述,在这种情况下这不会有任何不同。
OOME 是 1. JVM 在释放所有垃圾并压缩堆后没有足够大的可用连续空间 block 来存储新对象的结果。在这种情况下,时间并不是真正的因素。如果内存是可达的,则无论它存在多久都无法收集它。
因此,您可能会遇到问题的原因有很多。一个是您正在尝试创建一个非常大的对象(字符串允许您这样做)并且没有足够大的空闲堆 block 来容纳它。另一个原因是您用无法收集的对象填满了堆,因为您仍在引用它们。
在这种情况下,我认为问题在于您将所有字符串放入数组列表中,因此无法在方法执行结束之前收集它们。值得注意的是,substr 实际上创建了一个新字符串,该字符串引用了原始字符串的字符数组,因此不允许收集它。如果您想从一个大字符串中提取一小部分文本并丢弃其余部分,这可能会成为一个问题。我认为这不是问题,但还是很高兴知道。
减少内存的最有效方法是创建您自己的 Collection 类并在调用迭代器时生成每个排列。这将消除同时将所有排列存储在内存中的需要。如果有时间,我会发布一些示例代码。
编辑:我在不改变算法基本结构的情况下编写了一个示例。它应该在一个普通的 JVM 中运行,以获得比您想要等待完成的更大的输入。本质上,通过将所有结果放入 Arraylist,您的内存消耗会根据输入字符串长度以阶乘速率增长。通过消除结果的存储,以下方法的内存使用量相对于输入字符串长度呈线性增长。
这并不完美,还有一些问题留给读者去解决。提示:如果您创建一个带有空字符串的 Permutor 会发生什么?
正如有人提到的,StringBuffer 并不是这里的最佳选择,但使用子字符串和 concat 可能会做得更好:
current.substring(0, position).concat(last).concat(current.substring(position));
例子:
public class Example
{
public static void main(String... args) {
Permutor p = new Permutor("abcdefghijklmnopqrstuvwxyz");
System.out.println(p.size());
int i = 0;
for (String s : p) {
System.out.println(i++ + ": " + s);
}
}
public static int factorial(int n) {
int fact = 1;
for (int i = 1; i <= n; i++) {
fact *= i;
}
return fact;
}
public static class Permutor extends AbstractCollection<String>
{
private String characters;
public Permutor(String s)
{
characters = s;
}
@Override
public Iterator<String> iterator()
{
if (characters.length() == 1) {
return Collections.singleton(characters).iterator();
} else {
return new PermutingIterator(characters);
}
}
@Override
public int size()
{
return factorial(characters.length());
}
}
private static class PermutingIterator implements Iterator<String>
{
private final char last;
private final Iterator<String> inner;
private String current;
private int position;
PermutingIterator(String s)
{
int lastIndex = s.length() - 1;
this.inner = new Permutor(s.substring(0, lastIndex)).iterator();
this.last = s.charAt(lastIndex);
}
@Override
public boolean hasNext()
{
return inner.hasNext() || (current != null && position <= current.length());
}
@Override
public String next()
{
while(true) {
if (current != null && position <= current.length()) {
return new StringBuffer(current).insert(position++, last).toString();
} else if (inner.hasNext()) {
position = 0;
current = inner.next();
} else {
throw new IllegalStateException("no more permutations available");
}
}
}
@Override
public void remove()
{
throw new UnsupportedOperationException();
}
}
}
关于java - 避免 OutOfMemoryError,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28729617/
我正在尝试将用户提供的经纬度值与数据库中的经纬度值进行比较。如果它们在彼此半径 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++)
我是一名优秀的程序员,十分优秀!