【ACwing 93】【模版】非递归实现组合型枚举——模拟递归
(题面来自ACwing)
从 1~n 这 n 个整数中随机选出 m 个,输出所有可能的选择方案。
输入格式
两个整数 n,m ,在同一行用空格隔开。
输出格式
按照从小到大的顺序输出所有方案,每行1个。
首先,同一行内的数升序排列,相邻两个数用一个空格隔开。
其次,对于两个不同的行,对应下标的数一一比较,字典序较小的排在前面(例如1 3 5 7排在1 3 6 8前面)。
数据范围
n>0 ,
0≤m≤n ,
n+(n−m)≤25
此题的正解是dfs枚举,在之前的博客中有所提及。现在考虑用循环模拟机器递归的方式来做。
计算机维护系统栈来实现递归,栈中每个元素记录三个状态:当前函数的参数、上一层递归的位置(旧栈顶),以及递归完成后,上一层递归函数应当执行的下一条语句。我们模拟的栈的上一层栈顶位置是显然的,不需要记录第二个参数。
递归的过程是:每次将栈顶的元素出栈,确认它对应的状态执行到了哪一步,从这一步继续向下执行,直到返回或者遇到下一个递归调用。举个例子:下面是此题正解的dfs函数,我们可以把它分为3段,如代码所示。
- void dfs(int x) {
- //--------#0
- if (sel.size() > m || sel.size() + n - x + 1 < m)
- return;
- if (x == n + 1) {
- rep(0, m - 1)
- printf("%d ", sel[i]);
- puts("");
- return;
- }
- sel.push_back(x);
- //--------#0
- dfs(x + 1);
- //--------#1
- sel.pop_back();
- //--------#1
- dfs(x + 1);
- //--------#2
- return;
- }
基于这段函数体,我们用循环来模拟递归调用的过程。定义状态state包含的两个元素为当前函数参数x、当前函数的执行位置;注意这里的状态和系统栈中有差异。系统栈中记录的是返回后函数的执行位置,这里为了更直观地演示当前函数的执行而做了修改。首先将第一个状态{1, 0}入栈(对应主函数中的递归调用入口)。接下来进入循环体,我们在栈不空时循环执行以下语句:
1、出栈,记录栈顶参数x和当前执行位置cur_addr。
2、如果cur_addr为0,说明该函数从头开始执行,我们执行第0段函数体。执行完毕后,我们先入栈{x, 1}(表示再退栈到该状态时应当执行第1段函数体了),仔把下一个函数的参数x+1、下一个函数的执行位置(0)入栈。之后continue进入下一层循环;
3、如果cur_addr为1,说明该执行第二段函数体;这时我们先后入栈{x, 2}(返回到该状态时执行第2段函数体)、{x+1, 0}(下一层递归),继续循环。
4、如果cur_addr为2,当前函数已执行完毕,不用再次入栈。继续循环即可。
完整main函数代码:
- struct State {
- int x, addr;
- State(int a, int b):
- x(a), addr(b) {}
- };
- int main() {
- cin >> n >> m;
- stack<State> sta;
- sta.push(State(1, 0));//dfs(1);
- while (sta.size()) {
- int x = sta.top().x, cur_addr = sta.top().addr;
- sta.pop();
- switch (cur_addr) {
- case 0:
- if (sel.size() > m || sel.size() + n - x + 1 < m)
- continue;
- if (x == n + 1) {
- rep(0, m - 1)
- printf("%d ", sel[i]);
- puts("");
- continue;
- }
- sel.push_back(x);
- sta.push(State(x, 1));
- sta.push(State(x+1, 0));
- continue;
- case 1:
- sel.pop_back();
- sta.push(State(x+1, 0));
- }
- }
- return 0;
- }
算法进阶上给出的代码与系统栈的返回方式更加契合,这里一并给出。
- #include <iostream>
- #include <cstring>
- #include <cstdio>
- #include <vector>
- using namespace std;
- vector<int> chosen;
- int top, address, sta[100010], n, m;
- inline void call(int x, int ret_addr) { //模拟系统栈指令call(),记录每个状态的参数和返回语句位置
- int pre = top;
- sta[++top] = x;
- sta[++top] = ret_addr;
- sta[++top] = pre;
- }
- inline int ret() { //模拟指令return,退栈并返回应该执行的下一条语句
- int ret_addr = sta[top - 1];
- top = sta[top];
- return ret_addr;
- }
- int main() {
- cin >> n >> m;
- call(1, 0);
- while (top) {
- int x = sta[top - 2];
- switch (address) {
- case 0:
- if (chosen.size() > m || chosen.size() + (n - x + 1) < m) {
- address = ret();
- continue;
- }
- if (x == n + 1) {
- for (int i = 0; i < chosen.size(); ++i)
- printf("%d ", chosen[i]);
- puts("");
- address = ret();
- continue;
- }
- chosen.push_back(x);
- call(x+1, 1); //入栈下一个状态
- address = 0; //下一个函数从头执行
- continue;
- case 1:
- chosen.pop_back();
- call(x+1, 2); //入栈下一个状态
- address = 0;
- continue;
- case 2:
- address = ret(); //当前状态已执行完毕,返回
- }
- }
- return 0;
- }
【ACwing 93】【模版】非递归实现组合型枚举——模拟递归的更多相关文章
- AcWing 93. 递归实现组合型枚举
AcWing 93. 递归实现组合型枚举 原题链接 从 1~n 这 n 个整数中随机选出 m 个,输出所有可能的选择方案. 输入格式 两个整数 n,m ,在同一行用空格隔开. 输出格式 按照从小到大的 ...
- ACAG 0x02-8 非递归实现组合型枚举
ACAG 0x02-8 非递归实现组合型枚举 之所以专门来写这道题的博客,是因为感觉从最根本处了解到了递归的机器实现. 主要的就是两个指令--Call和Ret. Call指令会将返回地址入栈(系统栈) ...
- ACWing93.递归实现组合型枚举
题面 \93. 递归实现组合型枚举 从 1∼n 这 n 个整数中随机选出 m 个,输出所有可能的选择方案. 输入格式 两个整数 n,m ,在同一行用空格隔开. 输出格式 按照从小到大的顺序输出所有方案 ...
- 4位组合型Excel文档密码怎么破解
现代社会我们会遇到各种密码,很多的密码我们一段时间不用就不知不觉的忘记了.很多的excel用户就遇到过这种情况,这个时候我们就需要一款Excel密码破解工具.Advanced Office Passw ...
- RxJava 1.x 笔记:组合型操作符
最近去检查眼睛,发现度数又涨了,唉,各位猿多注意保护自己的眼睛吧! 前面学了 RxJava 的三种关键操作符: 创建型操作符 过滤型操作符 变换型操作符 读完本文你将了解第四种(组合型操作符): 组合 ...
- Kubernetes用户指南(二)--部署组合型的应用、连接应用到网络中
一.部署组合型的应用 1.使用配置文件启动replicas集合 k8s通过Replication Controller来创建和管理各个不同的重复容器集合(实际上是重复的pods). Replicati ...
- Untargeted lipidomics reveals specific lipid abnormality in nonfunctioning human pituitary adenomas 非靶向脂质组学揭示非功能人类脑垂体瘤中的特异性脂质 (解读人:胡丹丹)
文献名:Untargeted lipidomics reveals specific lipid abnormality in nonfunctioning human pituitary adeno ...
- AcWing 94. 递归实现排列型枚举
AcWing 94. 递归实现排列型枚举 题目链接 把 1~n 这 n 个整数排成一行后随机打乱顺序,输出所有可能的次序. 输入格式 一个整数n. 输出格式 按照从小到大的顺序输出所有方案,每行1个. ...
- java面试-公平锁/非公平锁/可重入锁/递归锁/自旋锁谈谈你的理解
一.公平锁/非公平锁/可重入锁/递归锁/自旋锁谈谈你的理解 公平锁:多个线程按照申请的顺序来获取锁. 非公平锁:多个线程获取锁的先后顺序与申请锁的顺序无关.[ReentrantLock 默认非公平.s ...
随机推荐
- 在 k8S 中搭建 SonarQube 7.4.9 版本(使用 PostgreSQL 数据库)
搭建 SonarQube 和 PostgreSQL 服务 本文搭建的 SonarQube 版本是 7.4.9-community,由于在官方文档中声明 7.9 版本之后就不再支持使用 MySQL 数据 ...
- Luogu P4546 [THUWC2017]在美妙的数学王国中畅游
题意 题意奇奇怪怪,这里就不写了. \(\texttt{Data Range:}1\leq n\leq 10^5,1\leq m\leq 2\times 10^5\) 题解 为什么你们都是卡在数学方面 ...
- DTU是怎么与PLC连接通信的
数据采集是生产制造中最实际最频繁的需求,不管智能设备制造发展到何种程度它都是工业4.0的先决条件,也在数字化工厂当中,工人更多地是处理异常情况,调整设备.但数据采集一直是困扰着所有制造工厂的传统痛点, ...
- 正式班D20
2020.11.02星期五 正式班D20 目录 11 软件包管理 11.1 软件包介绍 11.1.1 编程语言分类 11.1.2 三种安装包 11.2 rpm包管理 11.2.1 rpm包简介 11. ...
- Elasticsearch数据库 | Elasticsearch-7.5.0应用搭建实战
Elasticsearch 是一个可用于分布式以及符合RESTful 风格的搜索和数据分析引擎.-- Elastic Stack 官网 搭建Elasticsearch的"那些事儿" ...
- 第 2 篇:上手 Vue 展示 todo 列表
作者:HelloGitHub-追梦人物 追梦人物的 Vue 系列教程在他的博客已经全部更新完成,地址: https://www.zmrenwu.com/courses/vue2x-todo-tutor ...
- 老师问学生while判断
老师问学生,这道题你会做了吗?如果学生答"会了(y)",则可以放学.如果学生不会做(n),则老师再讲一遍,再问学生是否会做了...... (1)直到学生会为止,才可以放学. (2) ...
- C#练习题 if
提示用户输入用户名,然后再提示输入密码,如果用户名是"admin"并且密码是"888888",则提示正确,否则,如果用户名不是admin还提示用户用户名不存在, ...
- linux常用命令和系统基本目录
Linux 基础命令及基本目录 一.网卡 1.网卡配置文件路径 /etc/sysconfig/network-scripts/ifcfg-eth0 配置文件: TYPE=Ethernet # 以太 ...
- 通过JS判断当前浏览器的类型
通过JS判断当前浏览器的类型,对主流浏览器Chrome.Edge.Firefox.UC浏览器.QQ浏览器.360浏览器.搜狗浏览器的userAgent属性值来判断用户使用的是什么浏览器. 不同浏览器的 ...