递归与循环

众所周知,在一些问题中用递归会使得真个思路变得简单,比如说斐波拉契数列。但同时,简单的思路也会带来比较冗余的压栈开销问题。以下的题只是为了阐述递归分析问题的思路,不讨论复杂度。

1 用递归实现数组求和

int AddAll(int arr[],int begin,int length)   //求数组arr,从begin到结束的和
{
    if (begin == length) return 0;
    int sum = AddAll(arr, begin+1,length);
    return sum + arr[begin];
}

int main()
{
    int arr[] = {1,2,3,4,5,6 };
    int length = sizeof(arr) / sizeof(arr[0]);
    int sum = AddAll(arr, 0, length);
    cout << sum << endl;
    system("pause");
    return 0;
}

总结:在求类似家和问题时,递归会将整个数组分成一部分一部分,我们可以去求从某部分到某部分的和,任意位置都可以。

2 用递归判断字符串是否相等

bool IsEqual(string s1, string s2)
{
    int str1 = s1.length();
    int str2 = s2.length();
    if (str1 != str2) return false;
    if (str1 == 0) return true;
    std::string::iterator it1=s1.begin();
    std::string::iterator it2 = s2.begin();
    if (*it1 != *it2) return false;

    return IsEqual(s1.substr(1,str1), s2.substr(1,str2));
}

int main()
{
    string s1 = "hello world";
    string s2 = "hello worl";
    bool ret = IsEqual(s1, s2);
    cout << ret << endl;
    system("pause");
    return 0;
}

思路:在这个题中,我们需要判断两个字符串是否相等。按照逻辑我们依次比较就行,在这里需要考虑得程序出口可能就要包括以下几点:
1.如果两个字符串的长度不一样,直接return false;
2.当长度一样时,我们分别取每个字符串的首字母进行比较。当一样时,继续比较下一个字符到整个字符串结尾的那么长的字符串是否相等。依次递归进行比较。

递归与循环_第1张图片

总结:在这类问题中,递归所采用的思想就是把一个大问题不断细分,直到最后可以直接去返回的根源,然后一层一层进行返回。

3 在n个球中,任一取m个(不放回),求有多少种取法

int f(int n, int m)
{
    //a,b,c 
    //ab,bc,ac
    if (n < m) return 0;
    if (n == m) return 1;
    if (m == 0) return 1;

    //n个里有一个特殊的,取法划分:包不包括这个特殊的
    //要么在被取的范围里,要么就不在里面,两种情况
    return (f(n - 1, m - 1) + f(n - 1, m));
}

int main()
{
    int k = f(10, 3);
    cout << k << endl;
    system("pause");
    return 0;
}

思路:中学的时候我们都做过排列组合问题,这个也是一个典型排列组合问题。在这道题里,我们可以将其中三个球标记为特殊的球,那么每次取的时候就会有两种情况,一是包含这种特殊的球,另一种是不包含。如图。

递归与循环_第2张图片

所以我们用递归实现的时候需要考虑到这种情况,每次取球的时候都会发生这种情况,直到m=0。在0个球取0个,只有一种取法。

总结:递归去考虑排列组合问题的时候,一定要抓住特征,找到递归的出口。

4 求n个元素的全排列

void f(string arr,int k)   //当前的交换位置
{
    int len = arr.length();
    if (k == (len-1))
    {
        for (int i = 0; i < len; i++)
        {
            cout << arr[i] << " ";
        }
        cout << endl;
    }
    for (int i = k; i < len; i++)
    {
        swap(arr[k], arr[i]);//试探
        f(arr,k+1);
        swap(arr[k], arr[i]);//回溯
    }
}

int main()
{
    string arr= "abc";
    f(arr,0);
    system("pause");
    return 0;
}

思路:这个题和上面的题型是比较类似的,但考虑的情况是远比上面多。每一次交换后然后递归出一种情况,接着回溯,使字符串恢复到最初的样子。接着继续排列下一种情况。

你可能感兴趣的:(C语言)