csp信奥赛C++高频考点专项训练之字符串 --【字符串综合】:[NOIP 2004 普及组] FBI 树

csp信奥赛C++高频考点专项训练之字符串 --【字符串综合】:NOIP 2004 普及组 FBI 树

题目描述

我们可以把由 0 和 1 组成的字符串分为三类:全 0 串称为 B 串,全 1 串称为 I 串,既含 0 又含 1 的串则称为 F 串。

FBI 树是一种二叉树,它的结点类型也包括 F 结点,B 结点和 I 结点三种。由一个长度为 2 N 2^N 2N 的 01 串 S S S 可以构造出一棵 FBI 树 T T T,递归的构造方法如下:

  1. T T T 的根结点为 R R R,其类型与串 S S S 的类型相同;
  2. 若串 S S S 的长度大于 1 1 1,将串 S S S 从中间分开,分为等长的左右子串 S 1 S_1 S1 和 S 2 S_2 S2;由左子串 S 1 S_1 S1 构造 R R R 的左子树 T 1 T_1 T1,由右子串 S 2 S_2 S2 构造 R R R 的右子树 T 2 T_2 T2。

现在给定一个长度为 2 N 2^N 2N 的 01 串,请用上述构造方法构造出一棵 FBI 树,并输出它的后序遍历序列。

输入格式

第一行是一个整数 N ( 0 ≤ N ≤ 10 ) N(0 \le N \le 10) N(0≤N≤10),

第二行是一个长度为 2 N 2^N 2N 的 01 串。

输出格式

一个字符串,即 FBI 树的后序遍历序列。

输入输出样例 #1
输入 #1
复制代码
3
10001011
输出 #1
复制代码
IBFBBBFIBFIIIFF
说明/提示

对于 40 % 40\% 40% 的数据, N ≤ 2 N \le 2 N≤2;

对于全部的数据, N ≤ 10 N \le 10 N≤10。

思路分析

题目要求根据长度为 (2^N) 的 01 串构造 FBI 树,并输出后序遍历序列。

FBI 树的定义:

  • 全 0 串 → B 结点
  • 全 1 串 → I 结点
  • 既有 0 又有 1 → F 结点

构造方法递归:根结点为整个串的类型,若串长度 >1,则从中间分割为等长的左右子串,递归构造左右子树。

后序遍历顺序:左子树 → 右子树 → 根结点。

由于只需要后序遍历结果,无需显式建树,直接递归处理区间即可:

  1. 若区间长度为 1,根据字符直接输出 'B' 或 'I'。
  2. 否则,先递归左半区间,再递归右半区间,最后根据当前区间内是否同时存在 0 和 1 输出 'F'/'B'/'I'。
    复杂度 (O(N \cdot 2^N)),(N \le 10) 完全可行。

代码实现

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

int n, len; // n:指数, len:串长
string s; // 01串

void dfs(int l, int r) { // 递归处理区间[l,r]
    if (l == r) { // 叶子结点
        if (s[l] == '0') cout << 'B'; // 全0
        else cout << 'I'; // 全1
        return;
    }
    int m = (l + r) / 2; // 中间分割点
    dfs(l, m); // 左子树
    dfs(m + 1, r); // 右子树
    // 统计当前区间内是否有0和1
    bool has0 = false, has1 = false;
    for (int i = l; i <= r; ++i) {
        if (s[i] == '0') has0 = true;
        else has1 = true;
        if (has0 && has1) break; // 已同时存在
    }
    if (has0 && has1) cout << 'F'; // 混合
    else if (has0) cout << 'B'; // 全0
    else cout << 'I'; // 全1
}

int main() {
    cin >> n >> s; // 输入
    len = 1 << n; // 2^n
    dfs(0, len - 1); // 从整个串开始递归
    cout << endl; // 换行(洛谷一般允许末尾换行)
    return 0;
}

功能分析

  • 输入处理:读取整数 (N) 和长度为 (2^N) 的 01 串。
  • 递归遍历dfs(l, r) 模拟后序遍历,先递归左右子树,再输出当前结点类型。
  • 结点类型判定:通过扫描区间内字符判断是否同时存在 '0' 和 '1',从而输出 'F'、'B' 或 'I'。
  • 输出结果:输出后序遍历序列。

【完整系列请查看专栏】:
信奥赛C++普及组CSP-J一等奖通关刷题题单及题解:
https://blog.csdn.net/weixin_66461496/category_12673810.html 点击跳转


各种学习资料,助力大家一站式学习和提升!!!

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
int main(){
	cout<<"##########  一站式掌握信奥赛知识!  ##########";
	cout<<"#############  冲刺信奥赛拿奖!  #############";
	cout<<"######  课程购买后永久学习,不受限制!   ######";
	return 0;
}

【秘籍汇总】(完整csp信奥赛C++学习资料):

1、csp/信奥赛C++,完整信奥赛系列课程(永久学习):

https://edu.csdn.net/lecturer/7901 点击跳转

2、CSP信奥赛C++竞赛拿奖视频课:

https://edu.csdn.net/course/detail/40437 点击跳转

https://edu.csdn.net/course/detail/41081 点击跳转

3、csp信奥赛高频考点知识详解及案例实践:

CSP信奥赛C++动态规划:
https://blog.csdn.net/weixin_66461496/category_13096895.html点击跳转

CSP信奥赛C++标准模板库STL:
https://blog.csdn.net/weixin_66461496/category_13108077.html 点击跳转

信奥赛C++提高组csp-s知识详解及案例实践:
https://blog.csdn.net/weixin_66461496/category_13113932.html 点击跳转

4、csp信奥赛冲刺一等奖有效刷题题解:

信奥赛C++普及组CSP-J一等奖通关刷题题单及题解:
https://blog.csdn.net/weixin_66461496/category_12673810.html 点击跳转

信奥赛C++提高组csp-j初赛&复赛真题题解(持续更新): https://blog.csdn.net/weixin_66461496/category_12808781.html 点击跳转

信奥赛C++提高组csp-s初赛&复赛真题题解(持续更新):
https://blog.csdn.net/weixin_66461496/category_13125089.html 点击跳转

5、GESP C++考级真题题解:

GESP(C++ 一级+二级+三级)真题题解(持续更新):https://blog.csdn.net/weixin_66461496/category_12858102.html 点击跳转

GESP(C++ 四级+五级+六级)真题题解(持续更新):https://blog.csdn.net/weixin_66461496/category_12869848.html 点击跳转

GESP(C++ 七级+八级)真题题解(持续更新):
https://blog.csdn.net/weixin_66461496/category_13117178.html 点击跳转

· 文末祝福 ·

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
int main(){
	cout<<"跟着王老师一起学习信奥赛C++";
	cout<<"    成就更好的自己!       ";
	cout<<"  csp信奥赛一等奖属于你!   ";
	return 0;
}
相关推荐
小欣加油10 小时前
leetcode3751 范围内总波动值I
java·数据结构·c++·算法·leetcode
代码中介商10 小时前
C++左值与右值:核心判断法则详解
开发语言·c++
玖玥拾10 小时前
C/C++ 基础笔记(七)
c语言·c++
珊瑚里的鱼11 小时前
手撕单例模式中的饿汉模式和懒汉模式,懒汉模式还要再多加一个C++11版本的
开发语言·c++·单例模式
zh路西法11 小时前
【Linux 串口通信】基于 C++ 多线程的同步/异步串口实现
linux·运维·c++·python
不会C语言的男孩11 小时前
C++ Primer 第12章:动态内存
开发语言·c++
thisiszdy12 小时前
<C++> 浅拷贝与深拷贝
c++
2023自学中12 小时前
Linux虚拟机 CMakeLists.txt:x86 与 ARM 双架构编译脚本
linux·c语言·c++·嵌入式
眠りたいです12 小时前
现代C++:C++17中的新库特性
开发语言·c++·c++20·c++17
天若有情67313 小时前
【C++趣味实战】仿写Burp代理逻辑!自定义可控迭代器:拦截Intercept/放行Forward/重放Repeater全实现
java·开发语言·c++