函数的递归(Week 3)

什么是递归

  • 引入

    • 函数可以嵌套调用:无论嵌套多少层,原理都一样
    • 函数不能嵌套定义:不能在一个函数里再定义另一个函数,因为所有函数一律平等
    • 问题:一个函数能调用它自己吗?
  • 举递归调用的简单例子
#include<iostream>
using namespace std;
int fact(int n)
{
if(n == 1)
return 1;
else
return n*fact(n-1);
}
int main(){
cout<<fact(4)<<endl;
return 0;
}
  • 递归调用与普通的函数调用一样

    • 函数调用时,总是开辟新内存空间放调用的函数,递归调用同理

深入理解递归过程

#include<iostream>
using namespace std;
int recur()
{
char c;
c = cin.get();
if(c != '\n')
recur();
cout<<c;
return 0;
}
int main()
{
recur();
return 0;
}
//若输入:abc \n
//打印顺序:\n cba

递归的作用

用递归完成递推

  • 再一次切饼
#include<iostream>
using namespace std; int q(int n)
{
if(n == 0)
return 1;
else
return(n+q(n-1));
}
int main()
{
cout<<q(4)<<endl;
return 0;
}
//只需递推式和边际条件就能写出递归
  • 递归与递推

    • 不同

      • 递推的关注点放在起始点条件(i=0 ➡️ i=n)
      • 递归的关注点放在求解目标上(i=n ➡️ i=0)
    • 相同
      • 重在表现第i次与第i+1次的关系
  • 用递归实现递推

    • 优点

      • 让程序变得简明
    • 方法

      • 把关注点放在要求解的目标

        从而

      • 找到第n次做第n-1次做之间的关系

      • 确定第1次的返回结果

  • 斐波那契数列

#include <iostream>
using namespace std;
int f(int n)
{
if(n==1)
return 1;
if(n==2)
return 2;
else
return(f(n-1)+f(n-2));
}
int main()
{
cout<<f(4)<<endl;
}

模拟连续发生的动作

  • 进制转换
#include<iostream>
using namespace std;
void convert(int x)
{
if((x / 2) != 0)
{
convert(x / 2);
cout<<x % 2; //把cout放在递归调用之后,就可以逆序打印出来
}
else
cout<<x;
} int main()
{
int x;
cin >> x;
convert(x);
return 0;
}
  • 汉诺塔问题
#include <iostream>
using namespace std;
void move(int m, char x, char y, char z) //将m个盘子从A经过B移动到C
{
if(m == 1)
{
cout<<"把一个盘子"从<<x<<"移动到"<<z<<endl;
}
else
{
move(m-1,x,z,y);
cout<<"把一个盘子"从<<x<<"移动到"<<z<<endl;
move(m-1,y,x,z);
}
}
int main()
{
int n;
cout<<"请输入盘数n=";
cin >> n;
cout<<"在三根柱子上移动"<<n<<"只盘的步骤为:"<<endl;
move(n,'A','B','C');
return 0;
}
  • 思考的方法

    • 搞清楚连续发生的动作是什么(定义函数)
    • 搞清楚 不同次动作之间的关系(描述递归函数之间的关系)
    • 搞清楚边界条件是什么(递归退出的边际条件)

进行“自动的分析”

  • 放苹果

    • 把M个苹果放在N个同样的盘子里,允许有的盘子空着不放,问共有多少种不同的分法?
    • 注意:5,1,1和1,5,1是同一种放法
    • 输入:7,3
    • 假设:有一个函数f(m,n)能告诉我答案
      • 如果:n/盘子数 > M/苹果数: f(n>m)➡️f(m,n)=f(m,m)
      • 如果:M/苹果数>=n/盘子数
        • 有盘子空着:f(m,n) = f(m,n-1)
        • 没盘子空着:f(m,n) = f(m-n,n)
#include <iostream>
using namesapce std;
int count(int m,int n)
{
if( m<= 1 || n <=0) return 1;
if(m < n)
return count(m,m);
else
return count(m,n-1) + count(m-n,n)
}
void main()
{
int apples,plates;
cin >> apples >> plates;
cout << count(apples,plates);
}
  • 逆波兰表达式

    • 拟波兰表达式

      • 一种把运算符前置的算术表达式(2 + 3 ➡️ + 2 3)
      • 编写程序求解任意仅包含+ - * /四个运算符的逆波兰表达式
#include <iostream>
using namespace std;
double notation()
{
char str[10];
cin >> str;
switch(str[0])
{
case '+' return notation()+notation();
case '-' return notation()-notation();
case '*' return notation()*notation();
case '/' return notation()/notation();
default: return atof(str);
}
} int main()
{
cout<<notation();
return 0;
}
//注意这个题目的数据输入方式
  • 在以上复杂的场景中,递归帮我们进行了“自动分析”

    • 方法

      • 先假设有一个函数能给出答案
      • 在利用这个函数的前提下,分析如何解决问题
      • 搞清楚最简单的情况下,答案是什么

习题

Quiz1 单词翻转

#include<iostream>
using namespace std;
int i = 0;
char input[501];
int recur() {
char c = input[i];
i++;
if (c == ' ') {
return 1;//*句
}
if (c != ' '&&c != '\0') {
recur();
cout << c;
}
return 1; //**句
}
int main() {
cin.getline(input, 501);
while (input[i] != '\0') {
if (recur() == 1)//***句
cout << ' '; }
if (input[i] == '\0') {
cout << endl;//最后输出换行符
return 0;
}
}
//详解:
//以输入'Hello World'为例,句***刚被执行时,i=0,当被执行的这次return返回来时,i=6;句***再次被执行时,i=7,当被执行的这次return返回来时,i=8(因为只取了一个空格)
//遇到第一个空格时*句的return内容被"吞"掉了,所以需要**句的存在来补上这一个return
//最终的输出是'olleh dlrow',其中第一个空格来自**句,第二个空格来自*句

Quiz2 角谷猜想

#include <iostream>
using namespace std;
int JGG(int x)
{
if(x == 1)
{
cout<<"End"<<endl;
return 0;
}
else if(x % 2 == 0)
{
cout << x << "/2="<< x/2<<endl;
JGG(x/2);
}
else
{
cout << x <<"*3+1="<< (x*3+1)<<endl;
JGG(x*3+1);
}
return 0;
} int main()
{
int n;
cin >> n;
JGG(n);
return 0;
}

Quiz3 排队游戏

#include <iostream>
using namespace std;
char a[101]={'\0'};
int i=0,j=0; int cut()
{
int movement = 0;
int flag = 0;
for (int i = 0; i < 101; i++) {
if (a[i] != '\0')
flag = 1;
} if(flag == 0)
return 0;
for (int i = 0; i < 101; i++) {
if(a[i]!='\0')
{
for (int j = i+1; j < 101; j++) {
if(a[j] == '\0')
continue;
if(a[j] == a[i])
break;
if(a[j] != a[i])
{
cout<<i<<' '<<j<<endl;
a[i] = '\0';
a[j] = '\0';
movement = 1;
break;
}
}
if(movement == 1)
break;
} } flag = 0;
for (int i = 0; i < 101; i++) {
if (a[i] != '\0')
flag = 1;
}
if(flag == 1)
cut();
} int main()
{
cin.getline(a,100,'\n');
cut();
return 0;
}

Quiz4 扩号匹配问题

#include <iostream>
#include <stack> //双返回值的递归方法我实在有点看不懂(也不知道有没有非看懂不可的必要
//反正用堆栈就能很好的解决了……
using namespace std;
int main()
{
string str;
while(cin >> str) { char simple[101];
for (int i = 0; i < str.length(); i++) {
if (str[i] == '(')
simple[i] = '$';
else if (str[i] == ')')
simple[i] = '?';
else simple[i] = ' ';
} stack<char> s;
stack<int> index;
for (int i = 0; i < str.length(); i++) {
if (str[i] == '(') {
s.push('(');
index.push(i);
}
if (str[i] == ')') {
if (!s.empty() && s.top() == '(') //&&两边的两个条件必须且顺序不能改!!!!!
{
simple[index.top()] = ' ';
index.pop();
s.pop();
simple[i] = ' ';
} else {
s.push(')');
index.push(i);
}
}
} cout << str << endl;
for (int i = 0; i < str.length(); i++) {
cout << simple[i];
}
cout<<'\n'; }
return 0;
}

Coursera课程笔记----C程序设计进阶----Week 3的更多相关文章

  1. Coursera课程笔记----C程序设计进阶----Week 5

    指针(二) (Week 5) 字符串与指针 指向数组的指针 int a[10]; int *p; p = a; 指向字符串的指针 指向字符串的指针变量 char a[10]; char *p; p = ...

  2. Coursera课程笔记----C程序设计进阶----Week 4

    指针(一) (Week 4) 什么是"指针" 互联网上的资源--地址 当获得一个地址,就能得到该地址对应的资源,所以可以把"网址"称为指向资源的"指针 ...

  3. Coursera课程笔记----C程序设计进阶----Week 1&2

    C程序中的函数(Week 1&2) 函数 函数的定义 对函数的普遍认识:y=f(x) C语言中的常用函数: 平方根: r = sqrt(100.0) 底数x的y次幂:k = pow(x,y) ...

  4. Coursera课程笔记----C++程序设计----Week3

    类和对象(Week 3) 内联成员函数和重载成员函数 内联成员函数 inline + 成员函数 整个函数题出现在类定义内部 class B{ inline void func1(); //方式1 vo ...

  5. 操作系统学习笔记----进程/线程模型----Coursera课程笔记

    操作系统学习笔记----进程/线程模型----Coursera课程笔记 进程/线程模型 0. 概述 0.1 进程模型 多道程序设计 进程的概念.进程控制块 进程状态及转换.进程队列 进程控制----进 ...

  6. Coursera课程笔记----Write Professional Emails in English----Week 3

    Introduction and Announcement Emails (Week 3) Overview of Introduction & Announcement Emails Bas ...

  7. Coursera课程笔记----Write Professional Emails in English----Week 1

    Get to Know Basic Email Writing Structures(Week 1) Introduction to Course Email and Editing Basics S ...

  8. Coursera课程笔记----计算导论与C语言基础----Week 6

    理性认识C程序 导论(Week 6) 明确学习进度 讲课内容 感性➡️理性➡️函数➡️指针等 作业练习 初级阶段 ➡️正常作业练习 C语言的由来 程序设计语言的分类 低级语言之机器语言 0010101 ...

  9. Coursera课程笔记----计算导论与C语言基础----Week 4

    感性认识计算机程序(Week 4) 引入 编程序 = 给计算机设计好运行步骤 程序 = 人们用来告诉计算机应该做什么的东西 问题➡️该告诉计算机什么?用什么形式告诉? 如果要创造一门"程序设 ...

随机推荐

  1. 2019-07-31【机器学习】无监督学习之降维NMF算法 (人脸特征提取)

    代码 from numpy.random import RandomState #加载RandomState用于创建随机种子 import matplotlib.pyplot as plt from ...

  2. 造轮子:实现一个简易的 Spring IoC 容器

    作者:DeppWang.原文地址 我通过实现一个简易的 Spring IoC 容器,算是入门了 Spring 框架.本文是对实现过程的一个总结提炼,需要配合源码阅读,源码地址. 结合本文和源码,你应该 ...

  3. Python 编程环境搭建(Windows 系统中)

    由于大家普遍使用 Windows 系统,所以本文只介绍 Windows 系统中 Python 环境的安装. 在 Windows 中安装 Python 与安装普通软件没什么差别,下载所需版本的安装包后, ...

  4. stand up meeting 12/21/2015

    part 组员                工作              工作耗时/h 明日计划 工作耗时/h    UI 冯晓云  完成PDF UI主页面的页面切换功能,待完善    4  完善 ...

  5. 数据结构与算法--树(tree)结构

    树 二叉树 遍历原则:前序遍历是根左右, 中序遍历是左根右,后序遍历是左右根. 二叉搜索树 特点:对于树中的每个节点X,它的左子树中所有节点的值都小于X,右子树中所有节点的值都大于X. 遍历:采取二叉 ...

  6. 关于赋值的Java面试题

    面试题:(1) short s = 1:s = s + 1;(2) short s = 1;s += 1;问:上面两个代码有没有问题,如果有,哪里有问题? 答:(1) 第一个是错的,会报错损失精度,因 ...

  7. Java的自动装箱

    JDK5的新特性自动装箱:把基本类型转换为包装类类型自动拆箱:把包装类类型转换为基本类型 注意一个小问题: 在使用时,Integer x = null;代码就会出现NullPointerExcepti ...

  8. Flutter 分页功能表格控件

    老孟导读:前2天有读者问到是否有带分页功能的表格控件,今天分页功能的表格控件详细解析来来. PaginatedDataTable PaginatedDataTable是一个带分页功能的DataTabl ...

  9. 通过纯css实现图片居中的多种实现方式

    html结构: <div class="demo" style="width: 800px;height: 600px; border:1px solid #ddd ...

  10. 2019-2020-1 20199326《Linux内核原理与分析》第七周作业

    实验内容:分析Linux内核创建一个新进程的过程 初始化Menu Os,输入fork可以看到menuos触发了一个fork系统调用 再开一个shell,进入调试模式,设置几个断点sys_clone,d ...