Skip to content

Latest commit

 

History

History
316 lines (249 loc) · 8.96 KB

Lecture-1.2-递归的思想与应用.md

File metadata and controls

316 lines (249 loc) · 8.96 KB

递归的思想与应用

1 递归是数学上分而治之的思想

  • 将原问题分解为规模较小的问题进行处理
    • 分解后的问题与原问题类型完全相同,但规模较小
    • 通过小规模问题的解,能够轻易求得原问题的解
  • 问题的分解是有限的(递归不能无限进行)
    • 当边界条件不满足时,分解问题(递归继续进行)
    • 当边界条件满足时,直接求解(递归结束)

1.1 递归模型的一般表示方法

1525163251581

2 递归在程序设计中的应用

  • 递归函数
    • 函数体中存在自我调用的函数
    • 递归函数必须有递归出口(base case 基准情况),在 c 的递归函数中若无基准情况是毫无意义的,因为递归调用将重复进行直到基准情形出现。
    • 函数的无限递归将导致程序崩溃
    • 在设计递归程序时,同一问题的所有较小的实例均可以假设运行正确,递归程序只需要把这些较小问题的解(他们通过递归奇迹般地得到)结合起来而形成现行问题的解。其数学依据是归纳法
  • 递归的基本法则
    • 基准情形(base case)。你必须要有某些基准的情形,他们不用递归就能求解。
    • 不断推进(making progress)。对于那些需要递归求解的情形,递归调用必须能够朝着产生基准情形的方向推进。
    • 设计法则。假设所有的递归调用都能运行。
    • 合成效益法则(compond interest rule) 。在求解一个问题的同一实例时,切勿在不同的递归调用中做重复性的工作。这一条可以通过求斐波那契数量列的递归程序说明。可以查看 2.2 章节,当传入的 index >= 3 时,会执行如下的操作 fac(n - 1) + fac(n - 2);,而当调用 fac(n - 1)时其实已经计算了后面要用到的 fac(n - 2) ,但是这里却没有使用已经计算了一次的结果,造成了大量重复运算,导致程序效率低下。

2.1 递归求和

1525163382095

  • 代码实现:
unsigned int sum(int n) {
    if (n > 1)
        return n + sum(n - 1);
    else if (n == 1) {
        return 1;
    }
}

2.2 斐波那契数列

1525163413002

  • 代码实现:
unsigned int fac(int n) {
    if (n >= 3) {
        return fac(n - 1) + fac(n - 2);
    }

    if (n == 2)
        return 1;

    if (n == 1)
        return 1;
}

2.3 用递归的方法编写函数求字符串长度

1525163475986

  • 代码实现:
unsigned int get_strlen(char *s) {
    if (*s != '\0')
        return 1 + get_strlen(s + 1);
    if (*s == '\0')
        return 0;
}

2.4 单向链表的转置

1525184315854

  • 代码实现:
List reverse(List list) {
    if ((list == NULL || list->Next == NULL)) {
        return list;
    } else {
        List guard = list->Next;
        List ret = reverse(list->Next);
        guard->Next = list;
        list->Next = NULL;
        return ret;
    }
}

2.5 单向排序链表的合并

1525184468714

  • 代码实现:
List List_Merge(List L1, List L2) {
    if (L1 == NULL) {
        return L2;
    } else if (L2 == NULL) {
        return L1;
    } else if (L1->Data <= L2->Data) {
        List list_1 = L1->Next;
        List list = List_Merge(list_1, L2);
        L1->Next = list;
        return L1;
    } else {
        List list_2 = L2->Next;
        List list = List_Merge(list_2, L1);
        L2->Next = list;
        return L2;
    }
}

2.6 汉诺塔问题

1525436823759

1525436858718

  • 代码实现:
//汉诺塔问题:从 a 到 c, b 为中转站
void HanoiTower(int n, char a, char b, char c) {
    if (n == 1) {
        cout << a << "-->" << c << endl;
    } else {
        HanoiTower(n - 1, a, c, b);
        HanoiTower(1, a, b, c);
        HanoiTower(n - 1, b, a, c);
    }
}

2.7 全排列问题

1525441443912

  • 代码实现:
void permutation(char *s, char *e) {
    if (*s == '\0') {
        cout << e << endl;
    } else {
        int len = strlen(s);

        for (int i = 0; i < len; i++) {
            if ((i == 0) || (s[0] != s[i])) {
                swap(s[0], s[i]);
                permutation(s + 1, e);
                swap(s[0], s[i]);
            }
        }
    }
}

2.8 逆序打印单链表中的偶数结点

2.8.1 函数的调用过程回顾

  • 程序运行后有一个特殊的内存区供函数调用使用
    • 用于保存函数中的实参,局部变量,临时变量等
    • 从起始地址开始往一个方向增长(如:高地址 -> 低地址)
    • 有一个专用的指针标识当前已经使用的内存的“顶部”,及栈顶指针
  • 对函数栈使用的思考
    • 随着内存的使用或者释放,指针向下或者向上移动
    • 与特定函数调用相关的数据都存在各自函数的栈中,当 g() 函数返回,空间被释放后回到 f() 所在的栈,这个时候 f() 函数的栈数据没有变化。因此我们可以利用函数栈保存信息的特性来逆序递归的解决下面的问题。
    • 回溯算法的本质就是在栈上面作标记,以便于回退的时候使用。

1525596336625

2.8.2 逆序打印单链表分析

1525596423323

  • 题目分析

    • 递归出口是当 list 指向空的时候,函数直接返回
    • 当 list 不为空的时候递归调用反转打印函数,打印 list->next 的逆序,逐步减小问题规模
    • 当最终 list == NULL 的时候开始返回,利用栈中存储的信息来一步一步打印出链表中元素的值
  • 代码实现:

void list_attach(int data,List *pRear) {
    List P;

    P = (List) malloc(sizeof(struct Node));
    P->Data = data;
    P->Next = NULL;

    (*pRear)->Next = P; //让尾节点的指针指向新的节点
    *pRear = P;         //更新尾指针指向的位置
}

List create_list(int n) {

    List P, Rear, t;

    P = (List) malloc(sizeof(struct Node));
    P->Next = NULL;
    Rear = P;

    for (int i = 0; i < n; i++) {
        list_attach(i, &Rear);
    }

    t = P;
    P = P->Next;
    free(t);

    return P;
}

//递归实现的反向输出链表中的偶数
void r_print_even(List list) {
    if (list != NULL) {
        r_print_even(list->Next);

        if ((list->Data % 2) == 0) {
            cout << list->Data << endl;
        }
    }
}

2.9 八皇后问题

1525602421047

  • 题目分析

  • 代码实现:

  #include<iostream>
  using namespace std;
  static int g_chessboard[8] = { 0 }, gCount = 0;

  void print() //输出每一种情况下棋盘中皇后的摆放情况
  {
      for (int i = 0; i < 8; i++) {
          int inner;
          for (inner = 0; inner < g_chessboard[i]; inner++)
              cout << "0";
          cout << "#";
          for (inner = g_chessboard[i] + 1; inner < 8; inner++)
              cout << "0";
          cout << endl;
      }
      cout << "==========================\n";
  }

  int check_pos_valid(int loop, int value)    //检查是否存在有多个皇后在同一行/列/对角线的情况
          {
      int index;
      int data;
      for (index = 0; index < loop; index++) {
          data = g_chessboard[index];
          if (value == data)
              return 0;
          if ((index + data) == (loop + value))
              return 0;
          if ((index - data) == (loop - value))
              return 0;
      }
      return 1;
  }

  void eight_queen(int index) {
      int loop;
      for (loop = 0; loop < 8; loop++) {
          if (check_pos_valid(index, loop)) {
              g_chessboard[index] = loop;
              if (7 == index) {
                  gCount++, print();
                  g_chessboard[index] = 0;
                  return;
              }
              eight_queen(index + 1);
              g_chessboard[index] = 0;
          }
      }
  }

  int main(int argc, char*argv[]) {
      eight_queen(0);
      cout << "total=" << gCount << endl;
      return 0;
  }
  • 小结
    • 程序运行后的栈存储区专供函数调用使用
    • 栈存储区用于保存实参,局部变量,临时变量,寄存器等函数调用信息
    • 利用栈存储区能够方便的实现回溯算法
    • 八皇后问题是栈回溯的经典应用,仔细思考八皇后问题可以了解回溯算法的设计

3 总结

  • 递归是一种将问题分而治之的思想
  • 用递归解决问题首先要建立递归的模型
  • 递归解法必须要有边界条件,否则无解
  • 不要陷入递归函数的执行细节,学会通过代码描述递归问题,我们需要了解的是递归函数背后的递归模型,只要掌握了递归模型,递归函数仅仅是递归模型的描述而已。