Lesson 2 - 递推
2024年10月19日大约 5 分钟
算法思想
从已知的初始条件出发,依据递推关系,推出所求的结果。
递推题目需要满足的条件:
- 从已知条件到所求问题之间,总存在相互的联系。(递推关系)
- 递推要确定2点:
- 确定初始条件
- 确定递推关系(找规律)
顺推和逆推:
运输椰子参考程序
#include<bits/stdc++.h>
using namespace std;
int a[1100];
int main(){
a[0] = 64062; // 初始值
for(int i=1; i<=5; i++){ // 顺推
a[i] = a[i-1]/2 - 1; // 递推式
}
cout<<a[5];
return 0;
}
猴子吃桃参考程序
#include <iostream>
using namespace std;
int a[30];
int main() {
int n;
cin>>n;
a[n] = 1; // 初始值
for(int i=n-1; i>=1; i--){ // 逆推
a[i] = (a[i+1]+1)*2; // 递推式
}
cout<<a[1];
return 0;
}
常见递推模型
斐波那契数列类型
递推式子:long long
参考例题:
- 初始值
- 递推式本质:第
i
个月的兔子由来:第i-2
个月的兔子(因为两个月后会生产) + 第i-1
个月的兔子(大兔子和未长大的兔子)
兔子繁殖参考程序
#include<bits/stdc++.h>
using namespace std;
int a[50];
int main(){
int n;
cin>>n;
a[1]=1, a[2]=1;
for(int i=3; i<=n; i++){
a[i] = a[i-1] + a[i-2];
}
cout<<a[n];
return 0;
}
- 初始值
- 递推式本质:第
i
年的牛由来 = 第i-2
年的牛(因为两年后会成熟会生产) + 第i-1
年的牛(母牛和未成熟的小牛)
养牛场参考程序
#include<bits/stdc++.h>
using namespace std;
int a[30];
int main(){
int n;
cin>>n;
a[1]=1, a[2]=2;
for(int i=3; i<=n; i++){
a[i]=a[i-1]+a[i-2];
}
cout<<a[n]-1; // 题目问生产的小牛,减去最初的1头母牛
return 0;
}
- 初始值
- 递推式本质:第
i
阶台阶的走法 = 第i-2
阶台阶的走法(一次迈2个台阶可直接到达第i
阶)+ 第i-1
阶台阶的走法(一次迈1个台阶可直接到达第i
阶)
台阶问题参考程序
#include <iostream>
using namespace std;
int a[30];
int main() {
int n;
cin>>n;
a[1]=1, a[2]=2;
for(int i=3; i<=n; i++){
a[i]=a[i-1]+a[i-2];
}
cout<<a[n];
return 0;
}
- 初始值
- 递推式本质:长为
i
的墙面的铺设方法 = 长为i-2
的墙面的铺设方法(铺两块瓷砖横着放)+ 长为i-1
的墙面的铺设方法(铺一块瓷砖竖着放)
铺瓷砖参考程序
#include<bits/stdc++.h>
using namespace std;
int a[30];
int main(){
int n;
cin>>n;
a[1]=1; a[2]=2;
for(int i=3; i<=n; i++){
a[i]=a[i-1]+a[i-2];
}
cout<<a[n];
return 0;
}
其他变种模型
- 递推式
- 初始值
- 递推式本质:添加第
条直线,这条直线会和之前的 条直线相交,且这些交点会把新添加的直线分割成 段。每一段都会形成一个新的区域。因此,新的区域数会比原来的增加 个。因此,第 步的区域数 是前一步的区域数 加上新增的 个区域,即:
直线分割平面参考程序
#include<bits/stdc++.h>
using namespace std;
int n,a[60];
int main(){
cin>>n;
a[1] = 2; // 初始值:1条直线,2个平面
for(int i=2; i<=n; i++){
a[i]=a[i-1]+i;
}
cout<<a[n];
return 0;
}
练习
- 递推式
- 初始值
斐波那契数列参考程序
#include <iostream>
using namespace std;
long long a[100];
int main() {
int n;
cin>>n;
a[1]=1, a[2]=1;
for(int i=3; i<=n; i++){
a[i]=a[i-1]+a[i-2];
}
cout<<a[n];
return 0;
}
- 递推式
- 初始值
- 递推式本质:第
次折叠,都会让之前 次折叠产生的折横对称的生成 倍的折横,同时,每次在中间会再多产生 条新的折痕。
纸的折痕参考程序
#include<bits/stdc++.h>
using namespace std;
int a[30],n;
int main(){
cin>>n;
a[1]=1;
for(int i=2; i<=n; i++){
a[i] = a[i-1]*2+1;
}
cout<<a[n];
return 0;
}
- 递推式
- 初始值
- 递推式本质:第
次折叠,都会让之前 次折叠产生的折横对称的生成 倍的折横,同时,每次在中间会再多产生 条新的折痕。
Pell数列参考程序
#include<bits/stdc++.h>
using namespace std;
int a[1000010],k;
int main(){
a[1]=1,a[2]=2;
for(int i=3; i<=1000000; i++){
a[i] = 2*a[i-1]+a[i-2];
a[i] = a[i]%32767; // 数字过大,注意按照题目要求求模(取余)
}
cin>>k;
while(k--){ // 注意 k 组数据的输入,每输入1组数据,就输出1行答案
int x;
cin>>x;
cout<<a[x]<<endl;
}
return 0;
}