【问题标题】:Converting loop implementation of Pancake Sort to a recursive implementation将 Pancake Sort 的循环实现转换为递归实现
【发布时间】:2016-03-04 04:08:33
【问题描述】:

我正在学习递归并想将我的循环转换为递归函数?这段代码的正确答案应该是什么(假设我已经编写了翻转方法来反转数组中的元素)?谢谢,

/**
 * Sorts an array of integers by repeatedly reversing 
 * subranges within the array. Prints the flip sequence. 
 */ 
public static void  sort( int[] array)
{   
    int size = array.length;
    if (!Ordered(array, size)){
        for(int i = size-1; i > 0; i--)
        {
            int j = findMax(array, 0, i );
            int flipPosition;

            if( j != i )
            {
                if( j != 0 ) {
                    flip( array, 0, j );
                    flipPosition = size-j;
                    System.out.print( flipPosition + " " );
                }

                flip( array, 0, i );
                flipPosition = size-i;
                System.out.print( flipPosition + " " );
            }   
        }
    }
    System.out.println( 0 );
}

【问题讨论】:

  • 你想使用什么排序算法?
  • @Logan 我的方法使用煎饼排序,它在翻转数组之前识别数组中的最大数字,并从左到右按升序对数字进行排序。不知道怎么修改成递归函数。
  • 好递归不是在公园里散步。我主要用于排序的是一种称为快速排序的算法。我通过并找出我的“基本案例”是什么并从那里开始工作。
  • 在我学习递归时对我有帮助的东西:隔离你的基本案例和你的行动。您必须将定义作为您的基础(“基础案例”)。您的定义允许您“退出”递归。在其他情况下,您执行一项操作并再次调用自己以继续执行一项操作,直到达到基本情况。
  • 您想递归地解决NP-hard问题吗?

标签: java recursion


【解决方案1】:

如果这是家庭作业,我不想编写你的程序,或者如果这是个人的,我不想破坏你的乐趣,所以我在 Ruby 中实现了一个注释很多的递归解决方案。它基本上归结为将最大元素移动到数组末尾所需的翻转,然后将相同的逻辑应用于通过排除最大值创建的子数组。递归调用返回排序后的子数组,因此只需附加最大值并将结果传递回该行。递归的基本情况是当你找到一个元素时,只需返回它。

# Function to find the index of the max element in an array.
# In case of ties, returns the lowest index
def max_index(a)
  a.each_index.inject { |memo, i| a[i] > a[memo] ? i : memo }
end

def pancake_sort(a)
  # An array with 0 or 1 elements is sorted (trivially),
  # just return it.  This is the base case for the recursion.
  return a if a.length < 2

  # Find location of the max, express it as the n'th element
  n = max_index(a) + 1

  # Flip the stack of the first n elements (max is last)
  # to put the max at the front, concatenate it with the
  # rest of the array, then flip the entire result.  This
  # will put max at the end. However, don't bother with all
  # that flipping if max was already at the end.
  a = (a.take(n).reverse + a.drop(n)).reverse if n < a.length

  # Recursively apply the logic to the subarray that excludes
  # the last (max) element.  When you get back the sorted
  # subarray, tack the max back onto the end
  return pancake_sort(a.take(a.length - 1)) << a[-1]
end

# Create an array of 20 random numbers between 0 and 99    
ary = Array.new(20) { rand(100) }
# Display the result
p ary
# Display the result after sorting
p pancake_sort(ary)

# Sample output:
# [70, 19, 95, 47, 87, 49, 53, 8, 89, 33, 22, 85, 91, 87, 99, 56, 15, 27, 75, 70]
# [8, 15, 19, 22, 27, 33, 47, 49, 53, 56, 70, 70, 75, 85, 87, 87, 89, 91, 95, 99]

【讨论】:

  • 是的,我明白你在说什么,写下了我的代码,但它在某个地方仍然有问题,你能帮我看看吗? pastebin.com/qyYgDBkg
  • 如果您希望人们帮助您找出代码不起作用的原因,您需要在问题中包含代码。这意味着所有代码,包括您正在调用的小实用程序函数。
  • 对不起。只是觉得我的代码太长了。给你:pastebin.com/uQ3KXYXv。当我尝试运行测试时,它得到了 StackFlow 错误或 Java IndexOutOf Bound。
  • 这里不想刻薄,但我希望你意识到你正在试图转移问题。如上所述,您询问了如何使用递归完成您发布的迭代算法。现在您要问的是您没有提供的实现有什么问题,并且仍然不是问题的一部分。
  • 我继续查看您在 pastebin 上的代码。你有几个问题。首先,您的基本情况if (i &lt;= 0) 缺少return 语句,这就是导致堆栈溢出的原因。其次,您的Order() 支票在第一个元素上搞砸了。因为它不是真的需要,所以我只是检查了一下,你的代码成功地为我排序了测试用例。
猜你喜欢
  • 1970-01-01
  • 2021-07-20
  • 2019-05-14
  • 1970-01-01
  • 1970-01-01
  • 2020-09-05
  • 1970-01-01
  • 2014-10-01
相关资源
最近更新 更多