函数的递归(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. 【Java】用IDEA搭建源码阅读环境

    用IDEA搭建源码阅读环境 参考自CodeSheep的Mac源码环境搭建, https://www.bilibili.com/video/BV1V7411U78L 但是实际上在Windows搭建的差别 ...

  2. Daily Scrum 12/14/2015

    Progress: Dong&Minlong: 基于Oxford Speech API成功实现语音输入的功能,但由于服务器存在访问次数的限制(每分钟6次),所以暂不准备将此功能加入ALPHA版 ...

  3. windows UAC 提权实验(CVE-2019-1388)

    --------------------------------------------------------------------------------- 声明:本文仅做学习,实验主机为虚拟机 ...

  4. Python中的可视化神器!你知道是啥吗?没错就是pyecharts!

    pyecharts是一款将python与echarts结合的强大的数据可视化工具,本文将为你阐述pyecharts的使用细则 前言 我们都知道python上的一款可视化工具matplotlib,而前些 ...

  5. php 全局变量和超全局变量

    global 全局变量(在当前页面全部地方有效) $GLOBALS['name'] 超全局变量(在整个网站全部地方有效) 一个包含了全部变量的全局组合数组.变量的名字就是数组的键.还有_GET,_PO ...

  6. 小程序里json字符串转json对象需注意的地方

    一.JSON字符串转换为JSON对象 要使用上面的str1,必须使用下面的方法先转化为JSON对象: //由JSON字符串转换为JSON对象 var obj = eval('(' + str + ') ...

  7. Enjoy the game

    这真的只是一道规律题 我找到规律了但是因为数据太大了我超时了 我们现在来看一下这道题 牛牛战队的三个队员在训练之余会自己口胡了一些题当做平时的益智游戏.有一天牛可乐想出了一个小游戏给另外两名队员玩,游 ...

  8. MySql id 设定为主键不自增后,再给 sort 字段增加自增属性

    需求 id 已经被设置为主键,但是没有给它设置 自增 属性.sort 起到一个排序的作用,需要给它设置一个 自增 属性 加自增属性的前提 表中的属性没有增加自增 赋予自增属性的字段,必须带有 索引 S ...

  9. Nginx入门及如何反向代理解决生产环境跨域问题

    1.Nginx入门与基本操作篇 注:由于服务器是windows系统,所以本文主要讲解Nginx在windows下的操作. 首先下载Nginx 解压缩,我们所有的配置基本都在万能的 nginx/conf ...

  10. STL--priority_queue--自定义数据类型

    STL中priority_queue的声明模板有3个参数priority_queue<Type,Container,Functional>. 当使用的数据类型Type为自定义数据类型时有以 ...