Bootstrap

递归算法-基于归纳法


在递归算法的设计思想里面,可以把递归算法分为两种大的类型,一种是基于归纳法的递归,另一种是基于分治法的递归。前者是将数学里面的归纳法,最终归于一个基础项的计算思想应用到算法设计中而形成的,后者则是把一个问题分成多个子问题进行求解。本篇博文讨论前者-基于归纳法的递归算法。


1, 基本思想方法


对于一个问题规则为n的问题P(n),归纳法的思想方法是:

1)基础步:a1是问题P(1)的解。

2)归纳步:对所有的k,1<k<n,若b是问题p(k)的解,则P(b)是问题p(k+1)的解。其中P(b)是对b的某种运算或处理。


这个论断大家应该都熟悉,在高中数学里经常用归纳法证明一个数学问题,就是当n=k(=1,2)时,给定结论满足。假定当k时也满足,然后证明k+1时也满足,这个原理基本上是一样的。


2,递归算法的例子


一个最为大家所熟知的例子就是计算阶乘,如下所述:


2.1 计算阶乘函数n!

输入:n

输出:n!

int factorial(int n){
     if(n==0)
           return 1;
     else
           return n*factorial(n-1);
}

2.2 基于递归的插入排序

基于这个的思想进行递归:

1)基础步:当n=1时,数组只有一个元素,它已经是排序的。

2)归纳步:如果前面k-1个元素已经排序的,则插入第k个元素,只需要将第k个元素与前面的k-1个

输入:数组A[],数组的元素个数n;

输出:按递增顺序排序的数组A[];

void insert_sort_rec(int a[],int n){
      int k;
      int m;
      n=n-1;
      if(n>0){
             insert_sort_rec(a,n);
             m=a[n];
             k=n-1;
             while( (k>=0)&&a[k]>m)){
                 a[k+1]=a[k];
                 k=k-1;
              }
              a[k+1]=a
       }
}

3,排列问题的递归算法


该问题的描述是生成n个数的所有排列。用递归可以采取以下步骤:

1)数组第一个元素为1,即排列的第一个元素为1,生成后面的n-1个元素的排列。

2)排列的第一个元素为2,生成后面的元素排列。

就这样继续,最后排列的第一个元素为n。

然后对于第一个元素为1的排列,再假定第二个元素为2,生成后面n-2个元素的排列,假定第二个元素为3....,有:

基础步:k=1时,只有一个元素,已经构成一个排列。

归纳步:对任意的k(1<k<=n),如果可由算法完成k后面的k-1个元素排序,问题就得到解决。


输入:数组a[],数组的元素个数n, 当前递归层次完成排列的元素个数k.

输出:数组a[]的所有排列。


void perm(int a[],int k,int n){
     int i;
     if(k==1){
         for(i=0;i<n;i++)
              printf("%d-",a[i]);
     }else{
         for(i=n-k;i<n;i++){
              swap(a[i],a[n-k]);
              perm(a,k-1,n);
              swap(a[i],a[n-k]);
         }
     }
}


;