【AcWing】蓝桥杯辅导课-递归与递推

ops/2025/1/18 18:59:39/

目录

1、递归

1.1 递归实现指数型枚举

1.2 递归实现排列型枚举

1.3 递归实现组合型枚举

1.4 带分数

方法一

方法二

2、递推

2.1 简单斐波那契

2.2 费解的开关

2.3 翻硬币

2.4 飞行员兄弟

方法一

方法二


1、递归

递归就是在函数内部自己调用自己
我们以递归的形式来实现斐波那契数列:0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89

#include<iostream>
using namespace std;int fib(int n)
{if(n == 1) return 0;if(n == 2) return 1;return fib(n - 1) + fib(n - 2);
}

对于每一个递归,都可以画出一颗递归搜索树

有n步,每一步都会分成两个分支,所以时间复杂度是O(2^n)

1.1 递归实现指数型枚举

92. 递归实现指数型枚举 - AcWing题库

这道题中,n是15,所以我们可以猜测时间复杂度是O(2^n)或O(n*2^n)

这道题我们可以采用DFS来做,对于DFS来说,搜索顺序非常重要。找到一个顺序,可以不重不漏地把每种方案找出。在这道题中,我们可以开一个数组,数组的1-n位,依次考虑每个数选或不选

对于位置1, 2, 3, ..., n,都有选和不选两种情况,所以是2^n。并且还要将每种方案输出,所以,时间复杂度是O(n*2^n)

#include<iostream>
using namespace std;const int N = 20;int n, a[N];void dfs(int u)
{if(u > n){for(int i = 1;i <= n;i ++)if(a[i] == 1)cout << i << " ";cout << endl;return ;}a[u] = 2;dfs(u + 1);a[u] = 1;dfs(u + 1);
}int main()
{ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);cin >> n;dfs(1);return 0;
}

1.2 递归实现排列型枚举

94. 递归实现排列型枚举 - AcWing题库

这道题也是采用DFS做。
顺序1:依次枚举每个数放到那个位置
顺序2:依次枚举每个位置放那个数
我们采用的是顺序2来做

#include<iostream>
using namespace std;const int N = 15;int n, a[N];
bool st[N];void dfs(int u)
{if(u > n){for(int i = 1;i <= n;i ++)cout << a[i] << " ";cout << endl;return ;}for(int i = 1;i <= n;i ++)if(!st[i]){st[i] = true;a[u] = i;dfs(u + 1);st[i] = false;}
}int main()
{ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);cin >> n;dfs(1);return 0;
}

1.3 递归实现组合型枚举

93. 递归实现组合型枚举 - AcWing题库

排列是考虑顺序的,1 2 3和1 3 2是不一样的
组合是不考虑顺序的,1 2 3和1 3 2是一样的
所以,在递归实现组合型枚举时,需要规定一个顺序,这样在枚举时就可以保证不重复了,这里以升序为例。也就是说,对于1 2 3,1 3 2,2 1 3,2 3 1,3 1 2,3 2 1我们只枚举1 2 3即可
顺序:从前往后,依次枚举每个位置是几
为了有序,我们只需要时刻保持相邻两个位置的数有序即可
排列类型枚举需要一个bool类型的判重数组,而组合类型枚举不需要,因为只需要保证顺序即可

#include <iostream>using namespace std;const int N = 30;int n, m;
int way[N];void dfs(int u, int start) // u表示现在要填序列的第几位,start表示这一位可以从那个数开始填
{if (u == m + 1){for (int i = 1; i <= m; i ++ ) cout << way[i] << " ";cout << endl;return;}for (int i = start; i <= n; i ++ ){way[u] = i;dfs(u + 1, i + 1);way[u] = 0; // 恢复现场}
}int main()
{ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);cin >> n >> m;dfs(1, 1);return 0;
}

这样子仍然是不完美的,我们会发现,有些情况是需要剪枝的。若从start到n都选,仍然不够m个数,就没必要再向后进行了。即u-1+n-start+1<m

#include <iostream>using namespace std;const int N = 30;int n, m;
int way[N];void dfs(int u, int start) // u表示现在要填序列的第几位,start表示这一位可以从那个数开始填
{if (u + n - start < m) return;  // 剪枝if (u == m + 1){for (int i = 1; i <= m; i ++ ) cout << way[i] << " ";cout << endl;return;}for (int i = start; i <= n; i ++ ){way[u] = i;dfs(u + 1, i + 1);way[u] = 0; // 恢复现场}
}int main()
{ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);cin >> n >> m;dfs(1, 1);return 0;
}

1.4 带分数

1209. 带分数 - AcWing题库

方法一

我们可以使用排列型枚举的方式,枚举出1-9的所有排列,再从中选出3个数,试试能不能完成

#include<iostream>
using namespace std;const int N = 15;int n, ans, a[N];
bool st[N];int to_digit(int l, int r)
{int res = 0;for(int i = l;i <= r;i ++)res = res * 10 + a[i];return res;
}void dfs(int u)
{if(u == 10){for(int i = 1;i <= 6;i ++){int a = to_digit(1, i);if(a >= n) return ;for(int j = i + 1;j <= 8;j ++){int b = to_digit(i + 1, j);int c = to_digit(j + 1, 9);if(b % c == 0) // 必须要整除{int m = a + b / c;if(m == n) ans ++;}}}return ;}for(int i = 1;i <= 9;i ++)if(!st[i]){st[i] = true;a[u] = i;dfs(u + 1);st[i] = false;}
}int main()
{ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);cin >> n;dfs(1);cout << ans << endl;return 0;
}
// 123|456|789
// [1, i], [i + 1, j], [j + 1, 9]
方法二

在上面的方法中,我们是没用利用到n的

对于n = a + b / c,也就是c * n = c * a - b,我们可以只枚举出a和c,然后利用这个公式算出b。此时b是算出来的,而不是枚举出来的。在方法一中,如果a和c已经确定,b有3位数,需要枚举6次,而现在只需要1次

在代码实现中,我们是在a每一个叶子的地方去枚举c,在c每一个叶子的地方去计算b,然后判断是否成立。

判断b是否合法:
(1) 看b中是否有与a,c相同的数字,若某一位为0或已经出现过,返回false
(2) 看a,b,c的数字是否1-9都出现1次

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>using namespace std;const int N = 10;int n;
bool st[N], backup[N];
int ans;bool check(int a, int c)
{long long b = n * (long long)c - a * c;if (!a || !b || !c) return false; // a,b,c都不能为0memcpy(backup, st, sizeof st);while (b){int x = b % 10;     // 取个位b /= 10;    // 个位删掉if (!x || backup[x]) return false;backup[x] = true;}for (int i = 1; i <= 9; i ++ )if (!backup[i])return false;return true;
}void dfs_c(int a, int c)
{if (check(a, c)) ans ++ ;for (int i = 1; i <= 9; i ++ )if (!st[i]){st[i] = true;dfs_c(a, c * 10 + i);st[i] = false;}
}void dfs_a(int a)
{if (a >= n) return;if (a) dfs_c(a, 0);for (int i = 1; i <= 9; i ++ ) // 枚举当前这一位可以用那些数字if (!st[i]){st[i] = true;dfs_a(a * 10 + i);st[i] = false;}
}int main()
{cin >> n;dfs_a(0);cout << ans << endl;return 0;
}

2、递推

递归是把一个问题分成同类子问题
递推是先计算出子问题,再推出原问题
我们使用递推的方式计算出斐波那契额数列

#include<iostream>
using namespace std;const int N = 100;int dp[N];int fib(int n)
{dp[1] = 0, dp[2] = 1;for(int i = 3;i <= n;i ++) dp[i] = dp[i - 1] + dp[i - 2];return dp[i];
}

当然,也可以使用滚动数组的方式

#include<iostream>
using namespace std;int fib(int n)
{if(n == 1) return 0;if(n == 2) return 1;int a = 0, b = 1;for(int i = 3;i <= n;i ++) {int c = a + b;a = b;b = c;}return b;
}

2.1 简单斐波那契

717. 简单斐波那契 - AcWing题库

#include<iostream>
using namespace std;const int N = 50;int n, dp[N];int main()
{ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);cin >> n;dp[0] = 0, dp[1] = 1, dp[2] = 1;for(int i = 3;i < n;i ++) dp[i] = dp[i - 1] + dp[i - 2];for(int i = 0;i < n;i ++) cout << dp[i] << " ";cout << endl;return 0;
}

2.2 费解的开关

95. 费解的开关 - AcWing题库

在开关问题中,会有两条性质;
(1) 对于一种按的方式,顺序无所谓
(2) 求最优解时,每个位置最多按一次

在这道题中,我们会发现,当第一行的按法确定之后,后面4行的按法是确定的。因为这道题中按一个开关还会影响这个开关上下左右的一个开关,当第一行的按法确定之后,暗的位置只能通过第二行相应位置改变,后面的行也是以此类推。所以,我们只需要枚举第一行的所有情况,然后后面根据第一行的按法来按,然后看最后一行是否全亮即可,若全亮,证明这种按法是可行的

#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;int n;
char g[5][5], backup[5][5];
int dx[5] = {-1, 0, 1, 0, 0}, dy[5] = {0, 0, 0, -1, 1};void turn(int x, int y)
{for (int i = 0; i < 5; i ++ ){int a = x + dx[i], b = y + dy[i];if (a < 0 || a >= 5 || b < 0 || b >= 5) continue;   // 在边界外,直接忽略即可backup[a][b] ^= 1;}
}int main()
{ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);cin >> n;while(n --){int ans = 7;for(int i = 0;i <5;i ++)for(int j = 0;j < 5;j ++) cin >> g[i][j];for(int op = 0;op < 32;op ++){int step = 0;memcpy(backup, g, sizeof g);// op的二进制位是1就反转for(int i = 0;i < 5;i ++)if(op >> i & 1){turn(0, i);step ++;}// 第一行翻转完成,翻转剩下的4行// 是否翻转根据前一行同一位置判断for(int i = 0;i < 4;i ++)for(int j = 0;j < 5;j ++)if(backup[i][j] == '0'){turn(i + 1, j);step ++;}// 判断一下最后一行,若最后一行全亮,则这个第一行的翻转方案可行bool flag = false;for(int i = 0;i < 5;i ++)if(backup[4][i] == '0'){flag = true;break;}if(!flag) ans = min(ans, step);}if(ans > 6) cout << -1 << '\n';else cout << ans << '\n';}return 0;
}

2.3 翻硬币

1208. 翻硬币 - AcWing题库

这道题属于一种提醒:给定一个初始状态和一个目标状态,再给一些操作,问从初始状态最少需要几步操作能到目标状态。常常采用bfs来做,但bfs的时间复杂度较高,在这道题中并不合适

因为这道题每次都需要按两个,所以我们可以抽象成一个开关控制两个的形式


其中绿色的就代表开关。我们从第1个字符开始比较,一直到第n-1个字符,因为题目说了一定有解,所以最后一个不需要比较。当我们当前比较的字符不同时,就按一下能控制这个字符的开关(注意:这里每个开关也只能操作一次)
此时第一个字符不同,所以我们需要按一下第一个开关(这里只有第一个开关能操作第一个字符,所以我们只要从前向后枚举,不会出现我们当前正在比较的字符会被两个开关操作的情形)

重复这个步骤,直到第n - 1个

#include<iostream>
#include<string>
using namespace std;string s1, s2;int main()
{ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);cin >> s1 >> s2;int ans = 0;for(int i = 0;i < s1.size() - 1;i ++){if(s1[i] != s2[i]){ans ++;s1[i] = s2[i];if(s1[i + 1] == '*') s1[i + 1] = 'o';else s1[i + 1] = '*';}}cout << ans << '\n';return 0;
}

2.4 飞行员兄弟

116. 飞行员兄弟 - AcWing题库

这道题看起来是和费解的开关十分类似的,但是不能使用费解的开关中的方法,因为这道题操作一个把手后,不像费解的开关中一样只会影响上下左右相邻的4个开关,而是会影响同行同列的所有开关。同时我们发现,这道题的数据量较小,所以我们可以直接进行暴力枚举每一个位置变或不变。实现方法有两种

方法一

dfs

#include<iostream>
#include<algorithm>
#include<vector>
using namespace std;typedef pair<int, int> PII;const int N = 5;int ans = 20, m;
vector<PII> v; // 存放答案操作了那些位置void turn(int x, int y, vector<vector<char>>& g)
{if (g[x][y] == '+') g[x][y] = '-';else g[x][y] = '+';// 改变第x行for (int i = 0; i < 4; i++){if (i == y) continue;if (g[x][i] == '+') g[x][i] = '-';else g[x][i] = '+';}// 改变第y行for (int i = 0; i < 4; i++){if (i == x) continue;if (g[i][y] == '+') g[i][y] = '-';else g[i][y] = '+';}
}void dfs(int x, int y, vector<PII>& res, int step, vector<vector<char>>& g)
{if (step > ans) return;if (y == 4){x += 1;if (x == 4){bool flag = false;for (int i = 0; i < 4; i++)for (int j = 0; j < 4; j++)if (g[i][j] == '+'){flag = true;break;}if (!flag && step < ans){ans = step;v = res;}return;}else y = 0;}// 改变res.push_back({ x, y });turn(x, y, g);dfs(x, y + 1, res, step + 1, g);turn(x, y, g);res.pop_back();// 不改变dfs(x, y + 1, res, step, g);
}int main()
{ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);vector<vector<char>> g(4, vector<char>(4)); // 记录起始状态for (int i = 0; i < 4; i++)for (int j = 0; j < 4; j++) cin >> g[i][j];vector<PII> res; // 记录从起始状态到当前状态操作了那些位置dfs(0, 0, res, 0, g);cout << ans << '\n';for (int i = 0; i < v.size(); i++){cout << v[i].first + 1 << " " << v[i].second + 1 << '\n';}return 0;
}
方法二

也可以使用递推的做法

#include <cstring>
#include <iostream>
#include <algorithm>
#include <vector>#define x first
#define y secondusing namespace std;typedef pair<int, int> PII;const int N = 5;char g[N][N], backup[N][N];int get(int x, int y)
{return x * 4 + y;
}void turn_one(int x, int y)
{if (g[x][y] == '+') g[x][y] = '-';else g[x][y] = '+';
}void turn_all(int x, int y)
{for (int i = 0; i < 4; i ++ ){turn_one(x, i);turn_one(i, y);}turn_one(x, y);
}int main()
{for (int i = 0; i < 4; i ++ ) cin >> g[i];vector<PII> res;for (int op = 0; op < 1 << 16; op ++ ){vector<PII> temp;memcpy(backup, g, sizeof g);        // 备份// 进行操作for (int i = 0; i < 4; i ++ )for (int j = 0; j < 4; j ++ )if (op >> get(i, j) & 1){temp.push_back({i, j});turn_all(i, j);}// 判断所有灯泡是否全亮bool has_closed = false;for (int i = 0; i < 4; i ++ )for (int j = 0; j < 4; j ++ )if (g[i][j] == '+')has_closed = true;if (has_closed == false){if (res.empty() || res.size() > temp.size()) res = temp;}memcpy(g, backup, sizeof g);        // 还原}cout << res.size() << endl;for (auto op : res) cout << op.x + 1 << ' ' << op.y + 1 << endl;return 0;
}

http://www.ppmy.cn/ops/151160.html

相关文章

Linux系统编程:深入理解计算机软硬件体系和架构

一、硬件体系 首先我们要知道&#xff0c;我们最常见的计算机&#xff08;笔记本&#xff09;以及我们不常见的计算机&#xff08;服务器&#xff09;其实本质上都是一堆硬件的结合&#xff1a;cpu、网卡、显卡、内存、磁盘、显示器、键盘…… 但他们并不是毫无章法地放在一起…

Android 12.0 息屏休眠后立即启动屏保功能实现

1.前言 在12.0的系统rom定制化开发中,在进行一些关于自定义屏保功能中,产品需要再息屏休眠的时候启动屏保功能,接下来 就需要分析监听息屏休眠的时候,启动屏保功能实现 2.息屏休眠后立即启动屏保功能实现的核心类 frameworks/base/services/core/java/com/android/serve…

速通Docker === 目录挂载 卷映射

目录 目录挂载 1. 目录挂载的基本概念 2. 挂载命令 3. 配置Nginx启动页 4. 注意事项 卷映射 1. 创建数据卷 2. 运行Nginx容器并挂载数据卷 3. 查找数据卷的宿主机路径 4. 修改配置文件 5. 重启Nginx容器 6. 验证Nginx是否正常工作 Docker挂载总结 目录挂载 卷…

C++学习记录

本文章建立在已学C语言的基础上 第一阶段 生成随机数函数&#xff1a;rand()。rand()%100指的是生成0~99的随机数。这样生成的随机数每次都是一样顺序出现的&#xff0c;为了防止这个问题出现&#xff0c;我们可以使用随机数种子&#xff0c;如下代码 #include<iostream&…

【Uniapp-Vue3】vite.config中安装插件unplugin-auto-import自动导入vue和uniapp

对着项目右键-->使用命令行窗口打开所在目录&#xff0c;就会弹出终端 在终端中输入如下命令&#xff0c;后回车。 npm install unplugin-auto-import 在项目目录下创建vite.config.js 在vite.config.js文件中输入如下代码&#xff1a; import { defineConfig } from vi…

wow-agent---task2使用llama-index创建Agent

一&#xff1a;创造俩个函数&#xff0c;multiply和add作为fuction calling被LLM当做工具来使用&#xff0c;实现计算一个简单的计算题&#xff1a; from llama_index.llms.ollama import Ollama from llama_index.core.agent import ReActAgent from llama_index.core.tools …

如何制作符合自己设备的FLM下载算法

如何制作符合自己设备的FLM下载算法 --------以I.MXRT1062 QSPI FLAH为例&#xff08;串行qspi nor flash&#xff09; 本文介绍一种基于i.mxrt1062的外挂flah的qspi nor flash下载算法FLM的一种方法&#xff0c;Flash 编程算法是一种用于擦除或下载应用程序到 Flash 设备的软…

【开源免费】基于Vue和SpringBoot的夕阳红公寓管理系统(附论文)

本文项目编号 T 146 &#xff0c;文末自助获取源码 \color{red}{T146&#xff0c;文末自助获取源码} T146&#xff0c;文末自助获取源码 目录 一、系统介绍二、数据库设计三、配套教程3.1 启动教程3.2 讲解视频3.3 二次开发教程 四、功能截图五、文案资料5.1 选题背景5.2 国内…