转自:http://blog.csdn.net/monsterxd/article/details/8449005

/*
*  题意,求节点数为n的二叉树的所有形态,先要想个方式来唯一标示一棵二叉树
*
*  方法一:一个前序+一个中序,可以还原一棵唯一的二叉树,故使用【前序输出的字符串+中序输出的字符串】
*  来唯一标示一棵二叉树。
*
*  方法二:【将一颗二叉树逐层遍历,若节点不为空,则记为X,为空记为O,最终得到的序列可以唯一标示一颗二叉树。】
*
*  建树过程采用递归,对已经建成的树的部分,每次节点逐一判断其左右儿子是否可以插入,可以的话,则插入,然后递归。
*  直到数的大小达到n,则记录这棵树的唯一标示,然后返回!
*
*  output:  例如输入3,
*  则输出结果,对应的二叉树如下所示:  
*                  X               X              X                X                     X
*              X    O         X    O         O    X           O   X              X      X
*           X   O         O   X                  X   O           O   X        O  O  O  O
*        O   O               O   O           O   O                  O  O
*/

代码如下所示:

#include <iostream>
#include <string>
#include <map>
#include <vector>
using namespace std; int n; //二叉树有n个节点
map<string, int> m; struct Node{
Node * left;
Node * right;
}; void PreOut(Node * head, string &s) //先序输出,根左右
{
s += "X"; //根
if(head->left != NULL) //左
PreOut(head->left, s);
else
s += "O";
if(head->right != NULL) //右
PreOut(head->right, s);
else
s += "O";
} void MidOut(Node * head, string &s) //中序输出,左根右
{
if(head->left != NULL) //左
MidOut(head->left, s);
else
s += "O";
s += "X";
if(head->right != NULL) //右
MidOut(head->right, s);
else
s += "O"; } //逐层遍历二叉树
void Bfs(Node * head, string &s)
{
vector<Node *> vec;
vec.push_back(head);
for(int i=0; i<vec.size(); i++)
{
if(vec[i] != NULL)
{
s += "X";
vec.push_back(vec[i]->left); //左儿子入队,为NULL的时候也入队
vec.push_back(vec[i]->right); //右儿子入队
}
else
s += "O";
}
} /*
* head: 为整棵树的根节点
* arr: 为二叉树节点的数组
* now: 为当前已经建立的树的节点数
* total:为输入n,总共的节点数
* 建树时间复杂度: 卡特兰数复杂度 C(2*n,n)/(n+1)
*/
void BuildAllKindsTree(Node * head, Node * arr, int now, int total)
{
if(now == total) //成功建立一棵树
{
string s = "";
//PreOut(head, s);
//MidOut(head, s);
Bfs(head, s);
m[s]++;
return;
} for(int i=0; i<now; ++i) //当前建立的树已经有 now 个节点
{
if( arr[total - 1 - i].left == NULL ) //该节点左儿子位置可以链接个节点
{
arr[total - 1 - i].left = &arr[total - 1 - now];
BuildAllKindsTree(head, arr, now+1, total);
arr[total - 1 - i].left = NULL; //递归回溯
}
if( arr[total - 1 - i].right == NULL) //该节点右儿子位置可以链接个节点
{
arr[total - 1 - i].right = &arr[total - 1 - now];
BuildAllKindsTree(head, arr, now+1, total);
arr[total - 1 - i].right = NULL;
}
}
} void output()
{
cout<<"*****************************************************************"<<endl;
map<string, int>::iterator iter = m.begin();
for(; iter != m.end(); iter++)
cout<<iter->first<<endl;
cout<<n<<"个节点的时候,二叉树总共有"<<m.size()<<"种情况。"<<endl;
cout<<"*****************************************************************"<<endl;
} int main()
{
while(cin>>n)
{
m.clear();
Node * a = new Node[n]; //new出n个节点
for(int i=0; i<n; i++)
{
a[i].left = NULL;
a[i].right = NULL;
} BuildAllKindsTree(&a[n-1], a, 1, n); output();
delete []a;
}
}

【2013微软面试题】输出节点数为n的二叉树的所有形态的更多相关文章

  1. Java实现二叉搜索树的添加,前序、后序、中序及层序遍历,求树的节点数,求树的最大值、最小值,查找等操作

    什么也不说了,直接上代码. 首先是节点类,大家都懂得 /** * 二叉树的节点类 * * @author HeYufan * * @param <T> */ class Node<T ...

  2. 求二叉树第n层节点数

    在知乎看到今日头条的一个面试题“求二叉树第n层节点数”:https://zhuanlan.zhihu.com/p/25671699,想到了这样一个解法,欢迎大家交流 我的解法采用递归的思想,从0层开始 ...

  3. 1094. The Largest Generation (25)-(dfs,树的遍历,统计每层的节点数)

    题目很简单,就是统计一下每层的节点数,输出节点数最多的个数和对应的层数即可. #include <iostream> #include <cstdio> #include &l ...

  4. Spark中Task,Partition,RDD、节点数、Executor数、core数目的关系和Application,Driver,Job,Task,Stage理解

    梳理一下Spark中关于并发度涉及的几个概念File,Block,Split,Task,Partition,RDD以及节点数.Executor数.core数目的关系. 输入可能以多个文件的形式存储在H ...

  5. Spark中Task,Partition,RDD、节点数、Executor数、core数目(线程池)、mem数

    Spark中Task,Partition,RDD.节点数.Executor数.core数目的关系和Application,Driver,Job,Task,Stage理解 from:https://bl ...

  6. 确定BP神经网络中的节点数

    输入层 输入层节点数=输入向量维数 MNIST例子中,单张MNIST图片大小为28*28,reshape为一维数组,长度为784,所以输入层节点数为784: network = Network([78 ...

  7. 每日微软面试题——day 6(打印所有对称子串)

    每日微软面试题——day 6(打印所有对称子串) 分类: 2.数据结构与算法2011-08-14 14:27 9595人阅读 评论(15) 收藏 举报 面试微软string测试systemdistan ...

  8. 【剑指Offer面试题】 九度OJ1385:重建二叉树

    题目链接地址: pid=1385">http://ac.jobdu.com/problem.php?pid=1385 题目1385:重建二叉树 时间限制:1 秒内存限制:32 兆特殊判 ...

  9. hiho一下 第一百零七周 Give My Text Back(微软笔试题)

    题目1 : Give My Text Back 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 To prepare for the English exam Littl ...

随机推荐

  1. linux read和write函数

    原文出处:http://blog.chinaunix.net/space.php?uid=20558494&do=blog&id=2803003read函数是Linux下不带缓存的文件 ...

  2. ADODB.Connection 错误 '800a0e7a'。。

    今天帮同学调程序的时候发现的:错误提示如下: ADODB.Connection 错误 '800a0e7a' 未找到提供程序.该程序可能未正确安装. /hua1/manage/inc/conn.asp, ...

  3. HDU 4630 No Pain No Game 树状数组+离线查询

    思路参考 这里. #include <cstdio> #include <cstring> #include <cstdlib> #include <algo ...

  4. c# ffmpeg常用参数

    c#  ffmpeg常用参数 转换文件格式的同时抓缩微图: ffmpeg -i "test.avi" -y -f image2 -ss 8 -t 0.001 -s 350x240 ...

  5. NAND Flash【转】

    转自:http://www.cnblogs.com/lifan3a/articles/4958224.html 以Micron公司的MT29F2G08为例介绍NAND Flash原理和使用. 1. 概 ...

  6. leetcode:Plus One

    Given a non-negative number represented as an array of digits, plus one to the number. The digits ar ...

  7. 【温故知新】c#异步编程模型(APM)--使用委托进行异步编程

    当我们用到C#类许多耗时的函数XXX时,总会存在同名的类似BeginXXX,EndXXX这样的函数. 例如Stream抽象类的Read函数就有 public abstract int Read(byt ...

  8. hdu2847(暴力)

    去年看的一道题目,但是竟然傻傻的用dfs+循环链表去做. 简直傻到爆.  不过现在做这题还是想了好久而且还有好几次WA,其实这题还是很水的.直接暴力枚举就行了,枚举的前提是要算好复杂度, 可以知道的是 ...

  9. 【笨嘴拙舌WINDOWS】计时器精度

    WINDOWS的大多数系统并非实时操作系统,所以不能规定计算机在某个精确到纳秒的时间让计算机做某项任务,如果规定了时间WINDOWS也将需要在完成了线程调度后,经行任务执行! 也就是说,如果你的应用程 ...

  10. UVa 11624 (BFS) Fire!

    也是一个走迷宫的问题,不过又有了点变化. 这里迷宫里有若干把火,而且火每秒也是向四个方向蔓延的.问人是否能走出迷宫. 我用了两遍BFS,第一遍把所有着火的格子加入队列,然后计算每个格子着火的时间. 第 ...