PAT 1043 Is It a Binary Search Tree[二叉树][难]
1043 Is It a Binary Search Tree(25 分)
A Binary Search Tree (BST) is recursively defined as a binary tree which has the following properties:
- The left subtree of a node contains only nodes with keys less than the node's key.
- The right subtree of a node contains only nodes with keys greater than or equal to the node's key.
- Both the left and right subtrees must also be binary search trees.
If we swap the left and right subtrees of every node, then the resulting tree is called the Mirror Image of a BST.
Now given a sequence of integer keys, you are supposed to tell if it is the preorder traversal sequence of a BST or the mirror image of a BST.
Input Specification:
Each input file contains one test case. For each case, the first line contains a positive integer N (≤1000). Then Ninteger keys are given in the next line. All the numbers in a line are separated by a space.
Output Specification:
For each test case, first print in a line YES if the sequence is the preorder traversal sequence of a BST or the mirror image of a BST, or NO if not. Then if the answer is YES, print in the next line the postorder traversal sequence of that tree. All the numbers in a line must be separated by a space, and there must be no extra space at the end of the line.
Sample Input 1:
7
8 6 5 7 10 8 11
Sample Output 1:
YES
5 7 6 8 11 10 8
Sample Input 2:
7
8 10 11 8 6 7 5
Sample Output 2:
YES
11 8 10 7 5 6 8
Sample Input 3:
7
8 6 8 5 10 9 11
Sample Output 3:
NO
题目大意:如初一系列数,如果它是给出一棵二叉搜索树的前根遍历或者是它的左右子数反转的二叉树的前根便利(关键字会重复出现), 那么就输出Yes,并且输出这棵二叉树的后根遍历,如果不是输出No。
//建立一棵树,存储,使用数组存储。
代码来自:https://www.liuchuo.net/archives/2153
#include <cstdio>
#include <vector>
using namespace std;
bool isMirror;
vector<int> pre, post;
void getpost(int root, int tail) {
if(root > tail) return ;
int i = root + , j = tail;
if(!isMirror) {
while(i <= tail && pre[root] > pre[i]) i++;
//这个i一直循环到右子树的根节点,因为root的右子树是>pre[root]的。
while(j > root && pre[root] <= pre[j]) j--;
//这个是找到左子树前序遍历的最后一个节点。
} else {
while(i <= tail && pre[root] <= pre[i]) i++;
while(j > root && pre[root] > pre[j]) j--;
}
if(i - j != ) return ;
getpost(root + , j);//遍历左子树,
getpost(i, tail);//遍历右子树
post.push_back(pre[root]);//后根遍历放进来。
//当时叶节点的时候,会在入口处的if直接return了。
}
int main() {
int n;
scanf("%d", &n);
pre.resize(n);
for(int i = ; i < n; i++)
scanf("%d", &pre[i]);//输入前序。
getpost(, n - );//获取后序,
if(post.size() != n) {
isMirror = true;
post.clear();
getpost(, n - );
}
if(post.size() == n) {//如果是正常搜索二叉树的话,应该是=n的,有这么个规律在的。
printf("YES\n%d", post[]);//0直接在这里输出。
for(int i = ; i < n; i++)
printf(" %d", post[i]);
} else {
printf("NO");
}
return ;
}
//柳神的应该是能AC的代码中最精简的了,不用建树,厉害,之前也见到过这样的题目,应该加深一下,学习了。
//正常来说的思路就是,建树,所以参考了以下代码,也十分整洁:https://www.nowcoder.com/questionTerminal/8bcd661314744321b55dce1c1bfa8c54
//全是套路==
#include <cstdio>
#include <vector>
using namespace std;
struct Node{
int value;
Node *left, *right;//用的是指针哟。
}; void Insert(Node* &root, int data){//今天自己建树不成功才发现, 原来这位大佬传的是指针的引用,只传指针是不行的,学习了!2018-9-7
if(root == NULL){
root = new Node;//新指向一个节点。
root -> value = data;
root -> left = NULL;
root -> right = NULL;
return;
}
if(data < root->value) Insert(root->left, data);
else Insert(root->right, data);//由此看来>=root都是放在右子树的。
} void PreOrder(Node* root, vector<int>& v){
if(root == NULL) return;
v.push_back(root->value);//先访问根节点,再左右子树。
PreOrder(root->left, v);
PreOrder(root->right, v);
} void PreMirrorOrder(Node* root, vector<int>& v){
if(root == NULL) return;
v.push_back(root->value);
PreMirrorOrder(root->right, v);
PreMirrorOrder(root->left, v);
} void PostOrder(Node* root, vector<int>& v){//注意这里传了引用,其实也可以将其设置为全局变量。
if(root == NULL) return;
PostOrder(root->left, v);
PostOrder(root->right, v);
v.push_back(root->value);
} void PostMirrorOrder(Node* root, vector<int>& v){
if(root == NULL) return;
PostMirrorOrder(root->right, v);//既然右边小,那么就先访问右边
PostMirrorOrder(root->left, v);//再访问左边,形成的是和正常地是一样的序列。
v.push_back(root->value);
} int main(){
int n;
Node* s = NULL;
scanf("%d", &n);
vector<int> num, pre, preM, post, postM;
for(int i=; i<n; i++){
int data;
scanf("%d", &data);
num.push_back(data);
Insert(s, data);//使用指针传递,第一次就不是null了.
//此处建树最终建成的是一个标准的搜索二叉树。
}
PreOrder(s, pre);
if(num == pre){//判断两个向量是否相等,直接判断就可以了。
PostOrder(s, post);
printf("YES\n");
for(unsigned int i=; i<post.size(); i++){
printf("%d", post[i]);
if(i < post.size()-) printf(" ");
}
}
else{
PreMirrorOrder(s, preM);
if(num == preM){
PostMirrorOrder(s, postM);
printf("YES\n");
for(unsigned int i=; i<postM.size(); i++){
printf("%d", postM[i]);
if(i < postM.size()-) printf(" ");
}
}
else printf("NO\n");
}
return ;
}
//这个感觉是正常地思路。
1.首先按输入序列,建成一个标准的二叉搜索树,并且保存输入序列为num(题目中给的也是前序遍历)。
2.然后对其进行前序遍历,得到结果pre;
3.此时将pre与输入序列对比,如果=,那么就是正常的二叉搜索树
4.否则就可能是镜像或者完全不是两者。
5.此时对二叉树进行镜像前序遍历(因为镜像也就是将左右子树反转,那么此时访问根节点后,再访问建好的二叉树的右子树不就相当于对镜像进行前序遍历了吗)
6.得到的结果是preM,如果和num相同那么就是镜像的,然后对其进行后序镜像遍历输出
7.否则不是二者,输出no.
//厉害,学习了,建树的过程,以及前序和后序遍历,都明白了,多复习!
PAT 1043 Is It a Binary Search Tree[二叉树][难]的更多相关文章
- PAT 1043 Is It a Binary Search Tree (25分) 由前序遍历得到二叉搜索树的后序遍历
题目 A Binary Search Tree (BST) is recursively defined as a binary tree which has the following proper ...
- PAT 1064 Complete Binary Search Tree[二叉树][难]
1064 Complete Binary Search Tree (30)(30 分) A Binary Search Tree (BST) is recursively defined as a b ...
- PAT 1043 Is It a Binary Search Tree
#include <cstdio> #include <climits> #include <cstdlib> #include <vector> co ...
- 【PAT】1043 Is It a Binary Search Tree(25 分)
1043 Is It a Binary Search Tree(25 分) A Binary Search Tree (BST) is recursively defined as a binary ...
- PAT 甲级 1043 Is It a Binary Search Tree (25 分)(链表建树前序后序遍历)*不会用链表建树 *看不懂题
1043 Is It a Binary Search Tree (25 分) A Binary Search Tree (BST) is recursively defined as a bina ...
- PAT甲级:1064 Complete Binary Search Tree (30分)
PAT甲级:1064 Complete Binary Search Tree (30分) 题干 A Binary Search Tree (BST) is recursively defined as ...
- PAT 甲级 1043 Is It a Binary Search Tree
https://pintia.cn/problem-sets/994805342720868352/problems/994805440976633856 A Binary Search Tree ( ...
- PAT Advanced 1043 Is It a Binary Search Tree (25) [⼆叉查找树BST]
题目 A Binary Search Tree (BST) is recursively defined as a binary tree which has the following proper ...
- PAT题库-1064. Complete Binary Search Tree (30)
1064. Complete Binary Search Tree (30) 时间限制 100 ms 内存限制 32000 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHE ...
随机推荐
- Recurrent Neural Network Language Modeling Toolkit代码学习
Recurrent Neural Network Language Modeling Toolkit 工具使用点击打开链接 本博客地址:http://blog.csdn.net/wangxingin ...
- XML高速入门
XML是什么 Extensible Markup Language 自己定义标签: 用来数据传输: 可扩展标记语言,是一种类似超文本标记语言的标记语言. 与HTML的比較: 1.不是用来替代HTML的 ...
- java集合的中的集合关系实现或继承关系图
放在这儿一目了然.
- Tomcat服务器的搭建
一.Tomcat服务器端口的配置 Tomcat的所有配置都放在conf文件夹之中,里面的server.xml文件是配置的核心文件. 如果想修改Tomcat服务器的启动端口,则可以在server.xml ...
- linux下用gcc如何生成预处理、汇编等文件
[gcc -E test.c -o test.i------>预处理文件生成.i 文件.] 1.c语言程序生成过程 C语言程序的生成过程可以简单的分为:编辑.预处理.编译.汇编.链接五个阶断. ...
- Android 使用线性布局LinearLayout和Button实现一个点红块游戏
这个游戏的功能类似打地鼠. 项目地址:https://github.com/moonlightpoet/RedBlock 程序下载试玩地址:https://github.com/moonlightpo ...
- 安装安全狗后,MP4无法播放
- 百度地图地址查询API使用
aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAgsAAALxCAIAAABdNHLmAAAgAElEQVR4nOy9/VMbZ5rvnf/hbM7zy3 ...
- 我觉得epoll和select最大的区别
最近在用epoll,网速资料很多,大家都说epoll和select的区别比较大,而且select要不停遍历所有的fd,效率要低,而且fd有限制. 但是我认为二者最大的区别在于 先看代码 while ( ...
- Docker源码分析(九):Docker镜像
1.前言 回首过去的2014年,大家可以看到Docker在全球刮起了一阵又一阵的“容器风”,工业界对Docker的探索与实践更是一波高过一波.在如今的2015年以及未来,Docker似乎并不会像其他昙 ...