为了理解贪婪方法和动态编程等高级算法概念,首先需要精通递归。
我对递归比较陌生。每当提出问题时,首先要记住的是使用迭代的解决方案。尽管我知道递归方法意味着什么以及它是如何工作的,但很难以递归的方式思考。
请回答以下问题提供帮助:
1)任何迭代方法都可以用递归代替,反之亦然吗?
例如,如何以递归方式打印大小为n的数组中的元素?
for i 0 to n
Print a[i]
2)如何递归地解决问题?步骤是什么?是否有任何提示可以确定问题可以递归解决?
例如:如果要求打印出字符串的所有子字符串
INPUT: CAT
OUTPUT: CAT,CA,A,AT,T
我可以快速提出迭代方法。使用两个循环可以解决问题。
但递归地如何解决它。如何确定问题可以递归解决。
如果回答我的第一个问题是肯定的,使用两次递归而不是迭代可以解决我的问题吗?
3)任何人都可以建议我一些材料/资源来彻底理解递归的概念吗?
答案 0 :(得分:14)
有一种思考递归的方法,使其像迭代一样简单。
在迭代中我们有一个循环。可以认为它有4个部分:
根据某些“控制”数据决定继续或停止,作为合理条件进行评估。
完成工作的身体。有时候,身体会与下一部分结合在一起。
一种更改“控制”数据的方法。经常换柜台。
再次调用构造(在本例中为循环)的方法。在c风格的语言中,这是由for,while或do语法提供的。
在递归中,我们有一个函数(有时是几个)。他们有相同的4个部分:
根据某些“控制”数据继续或停止的决定被评估为逻辑条件。控制数据通常作为参数传递给函数。
完成工作的身体。有时候,身体会与下一部分结合在一起。
一种更改“控制”数据的方法。经常换柜台。
再次调用构造(在本例中为函数)的方法 - 这意味着调用函数(并记住传递更改的“控制”数据。
两个构造具有相同的部分应该不足为奇,因为它们是等价的。
答案 1 :(得分:7)
是的,主要是。一般来说,递归是为程序员而不是计算机进行的。迭代方法在某些情况下可能比递归方法运行得更快,但迭代方法可能需要300行代码和递归3.还有一些情况下,很容易弄清楚如何递归编程,但非常很难反复写,反之亦然。
一般来说,递归解决方案需要在功能方面。如果我们使用类似C ++的东西,我们可以使用处理字符串引用和事物的解决方案,慢慢调整作为参数传递的字符串。接近“两次递归”结束时的观点是错误的。这里的原则是,我们可以做一个递归方法而不是两次迭代。
http://introcs.cs.princeton.edu/java/23recursion/这个网站(谷歌搜索的最高点)讲授了许多递归的数学理论,并包含了一个常见问题解答,可能会给你一个更令人满意的答案。
答案 2 :(得分:5)
让我们完成一项简单的任务。打印从1到10的数字。我将在这里使用Python2.7。
for i in range(1,11):
print i
现在让我们尝试使用递归来做同样的事情。
>>> def print_me(n):
if n > 0:
print_me(n - 1)
print n
else:
return
>>> print_me(10)
1
2
3
4
5
6
7
8
9
10
那我们怎么想呢?
Step3 :我们首先编写一个执行打印功能的函数 数字作为参数传递。让我们想一想,主要的 任务。
def print_me(n): print n
第4步:我希望函数返回,如果n< 1。
def print_me(n): 如果n> 0: 打印 其他: 返回
第5步:现在我想将1到10之间的数字传递给此函数,但是 我们不想要一个1到10的循环,然后将它传递给我们的函数。我们 希望它以递归方式完成。
什么是递归? 简单来说,重复应用递归过程或定义。
因此,为了使其递归,我们需要调用函数本身。我们希望通过1到10的范围。
def print_me(n):
if n > 0:
print_me(n - 1)
print n
else:
return
<强>总结: 所有递归调用都必须遵守3条重要规则:
来源:interactivepython
Javascript中另一个找到阶乘的程序:
function factorial(n){
if (n == 1)
{
return 1;
}
else {
return n * factorial(n-1);
}
}
答案 3 :(得分:0)
@Test
public void testStrings() {
TreeSet<String> finalTree = getSubStringsOf("STACK");
for(String subString : finalTree){
System.out.println(subString);
}
}
public TreeSet<String> getSubStringsOf(String stringIn) {
TreeSet<String> stringOut = new TreeSet<String>();
if (stringIn.length() == 1) {
stringOut.add(stringIn);
return stringOut;
} else {
for (int i = 1; i < stringIn.length() ; i++) {
String stringBefore = stringIn.substring(0, i);
String stringAfter = stringIn.substring(i);
stringOut.add(stringBefore);
stringOut.add(stringAfter);
stringOut.addAll(getSubStringsOf(stringBefore));
stringOut.addAll(getSubStringsOf(stringAfter));
}
return stringOut;
}
}
我不知道你是否需要解释。每次可能的时候将字符串分成两部分。因此,Cat被分成CA,T和C,AT,您将它们添加到子字符串列表中,然后查找这些子字符串的每个子字符串。如果String是单个字符,则将单个字符添加到树中。
编辑:这是STACK的输出:
A
AC
ACK
C
CK
K
S
ST
STA
STAC
T
TA
TAC
TACK
再次编辑:正如您所看到的,每次运行方法subString时,您将在其中使用它两次,除非它是单个字符String。因此复杂度为O(n²)。对于'STACK',程序长度为0.200 ms,“STACKSTACKSTACK”(3次堆栈)为2秒,“STACKSTACKSTACKSTACKSTACK”理论上为2 ^ 10倍,因此为2000秒。
答案 4 :(得分:0)
这是我在Python中的简单测试:
def p(l, index):
if index == len(l):
return
else:
print l[index]
index = index + 1
p(l, index)
并致电:
p("123456", 0)
答案 5 :(得分:0)
以下是使用c ++递归打印数组的代码
#include<iostream>
using namespace std;
void print(int arr[],int n)
{
if(n==0) //base case. function call hits the base case when n==0
{
cout<<arr[0]<<" ";
return;
}
print(arr,n-1); //function will be called recursively until it reaches the
base case.
cout<<arr[n]<<" ";
}
//Driver function
int main()
{
int arr[]={10,20,30,40,50}; //array has been initialized with values
// 10,20,30,40,50
int n=sizeof(arr)/sizeof(arr[0]) ; //n stores the size of array,i.e, n=5;
print(arr,n-1); //print function has been called with arr[] and (n-1) as
parameters.
return 0;
}
我希望它能以某种方式提供帮助
答案 6 :(得分:0)
可以通过以下方式解决CAT问题:(Python代码)
s = "CAT"
op = []
def sub(s):
# terminating condition.
if (len(s) == 0) : return
if s not in op : op.append(s)
# slices from begning
sub(s[1:])
# slices from the end
sub(s[:-1])
sub(s)
print(op)
Output : ['CAT', 'AT', 'T', 'A', 'CA', 'C']