洛谷 B3622:枚举子集(递归实现指数型枚举)← DFS

【题目来源】
https://www.luogu.com.cn/problem/B3622

【题目描述】
今有 n 位同学,可以从中选出任意名同学参加合唱。
请输出所有可能的选择方案。

【输入格式】
仅一行,一个正整数 n。

【输出格式】
若干行,每行表示一个选择方案。
每一种选择方案用一个字符串表示,其中第 i 位为 Y 则表示第 i 名同学参加合唱;为 N 则表示不参加。
需要以字典序输出答案。

【输入样例】
3

【输出样例】
NNN
NNY
NYN
NYY
YNN
YNY
YYN
YYY

【数据范围】
对于 100% 的数据,保证 1≤n≤10。

【算法分析】
● DFS 算法模板:https://blog.csdn.net/hnjzsyjyj/article/details/118736059

cpp 复制代码
void dfs(int step) {
    判断边界 {
        输出解
    }

    尝试每一种可能 {
        满足check条件{
            标记
            继续下一步:dfs(step+1)
            恢复初始状态(回溯的时候要用到)
        }
    }
}

● 在算法层面,排列与组合问题本质是一类多分支、多阶段的枚举决策问题,需要在给定元素集合中,按照有序或无序的约束规则逐步筛选、分步构造合法方案。DFS 天然具备分层分支探索、深度遍历与状态回溯的固有特性,可逐层展开多路径选择,深入枚举每一种可行情况。在当前分支搜索完成后,通过回溯复原状态、切换新分支继续求解,天然适配多分支枚举场景,是解决此类多分支决策问题的核心算法。

【算法代码一】
● 这是一个标准的二进制递归生成器 ,通过深度优先搜索遍历所有可能的组合。它的核心思想是:在每个位置,我们有两种选择(N 或 Y),选择一种后递归处理下一个位置,返回后尝试另一种选择。这种模式是解决组合枚举问题的经典范式。

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;

int n;

void dfs(int pos,string s) {
    if(pos>n) {
        cout<<s<<endl;
        return;
    }
    dfs(pos+1,s+"N");
    dfs(pos+1,s+"Y");
}

int main() {
    cin>>n;
    dfs(1,"");
    return 0;
}

/*
in:
3

out:
NNN
NNY
NYN
NYY
YNN
YNY
YYN
YYY
*/

● 关键特性解析:dfs(pos, s) 函数中的 s 是按值传递的,这意味着每次递归调用都会创建 s 的一个副本。这天然实现了回溯。即在递归返回后,上一层的 s 保持不变。
例如:在 dfs(2, "N") 中调用 dfs(3, "NN") 后,返回到 dfs(2, "N") 时,s 仍然是 "N",而不是 "NN"。
模拟执行过程(以 n=3 为例),如下所示。

cpp 复制代码
初始调用:dfs(1, "")
├─ 选择 N (s="N") → dfs(2, "N")
│  ├─ 选择 N (s="NN") → dfs(3, "NN")
│  │  ├─ 选择 N (s="NNN") → dfs(4, "NNN") → 输出 "NNN" ✓
│  │  └─ 选择 Y (s="NNY") → dfs(4, "NNY") → 输出 "NNY" ✓
│  └─ 选择 Y (s="NY") → dfs(3, "NY")
│     ├─ 选择 N (s="NYN") → dfs(4, "NYN") → 输出 "NYN" ✓
│     └─ 选择 Y (s="NYY") → dfs(4, "NYY") → 输出 "NYY" ✓
└─ 选择 Y (s="Y") → dfs(2, "Y")
   ├─ 选择 N (s="YN") → dfs(3, "YN")
   │  ├─ 选择 N (s="YNN") → dfs(4, "YNN") → 输出 "YNN" ✓
   │  └─ 选择 Y (s="YNY") → dfs(4, "YNY") → 输出 "YNY" ✓
   └─ 选择 Y (s="YY") → dfs(3, "YY")
      ├─ 选择 N (s="YYN") → dfs(4, "YYN") → 输出 "YYN" ✓
      └─ 选择 Y (s="YYY") → dfs(4, "YYY") → 输出 "YYY" ✓

【算法代码二】

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;

int n;
string s;

void dfs(int pos) {
    if(pos>n) {
        cout<<s<<endl;
        return;
    }

    s+="N";
    dfs(pos+1);
    s.pop_back();

    s+="Y";
    dfs(pos+1);
    s.pop_back();
}

int main() {
    cin>>n;
    dfs(1);
    return 0;
}

/*
in:
3

out:
NNN
NNY
NYN
NYY
YNN
YNY
YYN
YYY
*/

● 关键特性解析

cpp 复制代码
初始调用:dfs(1),s=""
├─ 选择 N (s="N") → dfs(2)
│  ├─ 选择 N (s="NN") → dfs(3)
│  │  ├─ 选择 N (s="NNN") → dfs(4) → pos>3,输出 "NNN"
│  │  └─ 回溯删N,选择 Y (s="NNY") → dfs(4) → pos>3,输出 "NNY"
│  └─ 回溯删N,选择 Y (s="NY") → dfs(3)
│     ├─ 选择 N (s="NYN") → dfs(4) →  pos>3,输出 "NYN"
│     └─ 回溯删N,选择 Y (s="NYY") → dfs(4) →  pos>3,输出 "NYY"
└─ 回溯删N,选择 Y (s="Y") → dfs(2)
   ├─ 选择 N (s="YN") → dfs(3)
   │  ├─ 选择 N (s="YNN") → dfs(4) → pos>3,输出 "YNN"
   │  └─ 回溯删N,选择 Y (s="YNY") → dfs(4) → pos>3,输出 "YNY"
   └─ 回溯删N,选择 Y (s="YY") → dfs(3)
      ├─ 选择 N (s="YYN") → dfs(4) → pos>3,输出 "YYN"
      └─ 回溯删N,选择 Y (s="YYY") → dfs(4) → pos>3,输出 "YYY"

【参考文献】
https://blog.csdn.net/hnjzsyjyj/article/details/148295326
https://blog.csdn.net/hnjzsyjyj/article/details/160534326
https://blog.csdn.net/hnjzsyjyj/article/details/118736059
https://blog.csdn.net/hnjzsyjyj/article/details/156342794
https://blog.csdn.net/hnjzsyjyj/article/details/156341089
https://blog.csdn.net/hnjzsyjyj/article/details/128103062

相关推荐
smj2302_7968265210 小时前
解决leetcode第3934题最短唯一子数组
数据结构·python·算法·leetcode
iiiiyu10 小时前
面向对象和集合编程题
java·开发语言·前端·数据结构·算法·编程语言
变量未定义~10 小时前
最长回文子串
数据结构·算法
代码中介商10 小时前
AVL树:自平衡二叉搜索树的奥秘
数据结构
玛卡巴卡ldf11 小时前
【LeetCode 手撕算法】(多维动态规划)不同路径、最小路径和、最长回文子串、最长公共子序列、编辑距离
java·数据结构·算法·leetcode·动态规划·力扣
被AI抢饭碗的人11 小时前
算法:数据结构
数据结构·算法
淞綰12 小时前
c语言的练习-字符串的练习-寻找最长连续字符以及出现次数
c语言·数据结构·学习·算法·c语言的练习
qq_2965532712 小时前
[特殊字符] 搜索插入位置:从O(n)到O(log n)的优雅进化
数据结构·算法·面试·分类·柔性数组
凯瑟琳.奥古斯特13 小时前
力扣3654:二维矩阵连续空位统计
数据结构·数据库·算法·职场和发展
故事和你9113 小时前
洛谷-【图论2-2】最短路3
开发语言·数据结构·c++·算法·动态规划·图论