java - 快速排序 java.lang.StackOverflowError

标签 java stack-overflow

我正在尝试计算快速排序算法在已经排序的数组“中枢轴、第一个枢轴、随机枢轴”三种情况下的执行时间。

中枢轴和随机对于任何大小都可以正常工作,但如果大小大于 25000,第一个枢轴情况会导致 stackoverflow。

代码如下:

static void q_sort( int left, int right)
{

int pivot;
int l_hold, r_hold;

  l_hold = left;
  r_hold = right;
  pivot = array[left];
  while (left < right)
  {
    while ((array[right] >= pivot) && (left < right))
      right--;
    if (left != right)
    {
      array[left] = array[right];
      left++;
    }
    while ((array[left] <= pivot) && (left < right))
      left++;
    if (left != right)
    {
      array[right] = array[left];
      right--;
    }
  }
  array[left] = pivot;
  pivot = left;
  left = l_hold;
  right = r_hold;
  if (left < pivot)
    q_sort(left,(int) pivot-1);
  if (right > pivot)
    q_sort( (int)pivot+1, right);
}

这是调用代码:

double startTime1 = System.currentTimeMillis();
q_sort(0,size -1);
double stopTime1 = System.currentTimeMillis();
double elapsedTime1 = stopTime1 - startTime1;      
System.out.println(elapsedTime1);

最佳答案

你没有告诉数组是如何生成的,但我怀疑它已经排序了。

问题如下:如果对已经排序的数组进行快速排序,第一个主元会导致以下递归:0..24999, 1..24999, 2..24999, 3..24999, 4..24999占用大量堆栈空间并导致堆栈溢出。这是因为数组为 0..24999,主元为 0,那么第二个分区将变为 1..24999。

您应该消除其中一个尾部调用。要消除哪个尾调用取决于哪个分区较小。您希望递归处理尽可能少的数据。其中一个递归被简单地替换为循环。这种尾递归消除已经在 Quicksort and tail recursive optimization 中进行了解释。

无论如何,我建议使用现有的快速排序算法,而不是自己编码,除非这是一个家庭作业问题。

关于java - 快速排序 java.lang.StackOverflowError,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29240519/

相关文章:

java - java中如何将多维列表转换为多维数组?

java - 为什么 lookingAt 为 lookArounds 返回 true 而 matches 返回 false

java - 在 Java 中,不可序列化的套接字是否必须阻止使用客户端-服务器观察者模式?

javascript - 如何防止由于不需要的递归而导致 IE 中的堆栈溢出

c# - 编译器是否可以创建具有 StackOverflowException 的异步代码?

java - Java中多重继承的替代方案

java - 通过避免在特定场景中重写 java 父类(super class)方法来重新使用父类(super class)方法

如果类的成员,C++ 数组会导致崩溃

c++ - 为什么 cout 的访问冲突和 printf 的堆栈溢出

java - 如何修复: Sudoku solver Stack overflow problem