函数的递归(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. intellij idea 设置用真机测试android

    android自带的模拟器是不容置疑的慢,genymontion虽然快,但是觉得有点怪的感觉,哈哈,其实这些都不是重点. 之前是用myeclipse开发android的,虽然一直很想用eclipse来 ...

  2. stand up meeting 12/01/2015

    part 组员 今日工作 工作耗时/h 明日计划 工作耗时/h UI 冯晓云   赶工sprint3,各部分要合在一起时出现各种问题,各种修改测试:UI本身的功能继续实现完善    6 UWP对控件的 ...

  3. E - Dividing Chocolate ATcoder

    题目大意:切割图形,给你一个非0即1的矩阵,将它切割成多个长方形,使每个小长方形中1的个数不得多于k个,切割的规则,要么切一整行,要么是一整列. 题解: 二进制枚举. 注意行数最大才是10.用二进制枚 ...

  4. E1. Send Boxes to Alice (Easy Version)

    题解: 保存每个1的位置.然后记录1的总个数cnt,如果存在一个k使得这个k是每个集合的倍数,那么为了使操作次数最小,这个k应该是cnt的质因子.(因为都是每个集合的数目1,使每个集合的数目变为2需要 ...

  5. SpringBoot 集成 Elasticsearch

    前面在 ubuntu 完成安装 elasticsearch,现在我们SpringBoot将集成elasticsearch. 1.创建SpringBoot项目 我们这边直接引入NoSql中Spring ...

  6. Mac安装aws-cli全过程,通过命令行上传文件到aws s3协议服务器

    第一次使用aws,首先查询了各种资料,我第一步需要做的是安装aws-cli,而安装aws-cli之前需要安装python3,当然你安装python3之前你还需要安装homebrew,当然我正在安装的过 ...

  7. RabbitMQ Hello world(二)

    简介: Rabbitmq 是消息代理中间件,它接收或者发送消息.你可以把它想想宬一个邮局:当你把邮件放到邮箱时,你可以确定某一位邮递员可以准确的把邮件送到收件人手中,在这个比喻中,rabbitmq是一 ...

  8. 二进制安装MySQL及破解密码

    二进制安装MySQL及破解密码 1.确保系统中有依赖的libaio 软件,如果没有: yum -y install libaio 2.解压二进制MySQL软件包 tar xf mysql-5.7.24 ...

  9. Uva 1754 Posterize

    #include<bits/stdc++.h> using namespace std; #define rep(i,a,b) for(int i=a;i<=b;++i) #defi ...

  10. Guzzle 一个PHP的HTTP客户端

    Guzzle是一个PHP的HTTP客户端,用来轻而易举地发送请求,并集成到我们的WEB服务上. https://github.com/guzzle/guzzle 接口简单:构建查询语句.POST请求. ...