gpt4 book ai didi

java - 概率计算的堆栈溢出异常

转载 作者:行者123 更新时间:2023-11-30 04:03:56 25 4
gpt4 key购买 nike

我正在做 Project Euler 上的第 14 题(注意:我不是在寻找 Project Euler 问题的解决方案)当我遇到一个有趣的堆栈溢出异常时。

我的非概率方法工作得很好,但是当我尝试使用概率方法解决同样的问题时,我遇到了堆栈溢出异常。有趣的是,异常只发生大约 17% 的次数。一千次运行产生了 166 个异常(exception)。

我知道我的概率逻辑有缺陷,但我更感兴趣的是异常的原因以及防止异常发生的方法。我是否只需要进行一些内存管理,也许在使用某些变量后将它们设置为空?如果是的话,关键点在哪里?

代码如下:

public class Problem14_LongestCollatzSequence {

private static final int STARTING_CHAIN_LENGTH = 1;
private static final int PROBABLY_RIGHT = 100000;

/**
* Calculate and return the Collatz sequence of a given number.
*
* @param number The number for which the Collatz sequence is to be
* calculated.
* @param chainlength The length of the chain for the number. This should
* start with an initial value of 1.
* @return The Length of the Collatz sequence.
*/
private static int getChainLength(long number, int chainlength) {
// All chains should end with 1.
if (number != 1) {
// If the number is even, halve the number, otherwise multiply it by 3 and add 1.
if (number % 2 == 0) {
number = number / 2;
} else {
number = number * 3 + 1;
}
// Call this function again.
return getChainLength(number, ++chainlength);
}
// Return the length of the chain.
return chainlength;
}

/**
* Determine and return the number below a maximum value that will result in
* the longest Collatz chain.
*
* @param maxStartingNumber The maximum value (exclusive) of the numbers
* that will be tested.
* @return The number that will produce the longest Collatz sequence in the
* given range.
*/
private static int calculateLongestChain(int maxStartingNumber) {
Random random = new Random();
int probabilityCounter = 0;
int currentChainNumber = 0;
int longestChainNumber = 0;
int currentChainLength = 0;
int longestChainLength = 0;

// Get the chain length of random numbers until a certain number of unsuccsessful attempts have been made.
while (probabilityCounter <= PROBABLY_RIGHT) {
currentChainNumber = random.nextInt(maxStartingNumber);
currentChainLength = getChainLength(currentChainNumber, STARTING_CHAIN_LENGTH);
// If the current chain-length is bigger than the previously calculated one, reset the counter and update the chain number, otherwise increase the counter.
if (currentChainLength > longestChainLength) {
probabilityCounter = 0;
longestChainLength = currentChainLength;
longestChainNumber = currentChainNumber;
} else {
++probabilityCounter;
}
}
return longestChainNumber;
}

private static int calculateLongestChainNP(int maxStartingNumber) {
// Non-probabilistic way to calculate the longest Collatz sequence.
int currentChainLength = 0;
int longestChainLength = 0;
int longestChainNumber = 0;
// Simply loop through all the numbers in the range to calculate the one resulting in the longest sequence.
for (int i = 1; i < maxStartingNumber; i++) {
currentChainLength = getChainLength(i, STARTING_CHAIN_LENGTH);
if (currentChainLength > longestChainLength) {
longestChainLength = currentChainLength;
longestChainNumber = i;
}
}
return longestChainNumber;
}

public static void main(String[] args) {
int exceptionCount = 0;
for (int count = 0; count < 1000; count++) {
try {
int testNumber = 1000000;
System.out.println("Probabilistic answer: " + calculateLongestChain(testNumber));
System.out.println("Non-probabilistic answer: " + calculateLongestChainNP(testNumber) + "\n");
} catch (java.lang.StackOverflowError soe) {
exceptionCount++;
System.err.println(soe + "\n");
}
}
System.out.println("Exception count: " + exceptionCount);
}
}

我也想提供完整的输出,但这超出了字符限制。

最佳答案

你的递归太深了。您可以使用 -Xss 4096m 增加 JVM 上的调用堆栈,但这是蛮力。更优雅地在 getChainLength() 中使用 while 循环而不是递归:

private static int getChainLength(long number, int chainlength) {
// All chains should end with 1.
while (number != 1) {
// If the number is even, halve the number, otherwise multiply it by 3 and add 1.
if (number % 2 == 0) {
number = number / 2;
} else {
number = number * 3 + 1;
}
// Call this function again.
++chainlength;
}
// Return the length of the chain.
return chainlength;
}

关于java - 概率计算的堆栈溢出异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21230710/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com