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 ...
随机推荐
- day04<Java语言基础+>
Java语言基础(循环结构概述和for语句的格式及其使用) Java语言基础(循环结构for语句的练习之获取数据) Java语言基础(循环结构for语句的练习之求和思想) Java语言基础(循环结构f ...
- 我们复习.Net的这些日子里
今天是我们复习前面学习的第三天,也是在今天我们结束了复习.之前上学时间感觉知识全是生僻的,在生活中很少用到,因此喜欢重复的记忆,而现在学习的知识全是现在乃至未来,将是每天我们使用的,所以就感觉没必要重 ...
- C++中的字节对齐分析
struct A { int a; char b; short c; }; struct B { char a; int b; short c; }; #pragma pack(2) struct C ...
- js方法区分IE浏览器和非IE浏览器
可以从IE特有的方法和非IE特有的方法来区分不同的浏览器 1.为元素添加事件监听: 非IE:.addEventListener("click",show,false)//第三个参数 ...
- Mybatis中的foreach
<delete id="deleteByParam"> DELETE FROM YZ_SECURITIES_CURRENCY WHERE ID IN <forea ...
- PHP之ob_start()控制缓冲生成html
输出控制函数不对使用 header() 或 setcookie(), 发送的文件头信息产生影响,只对那些类似于 echo() 和 PHP 代码的数据块有作用. 我们先举一个简单的例子,让大家对Outp ...
- poj_2553 强连通分支&出度为0的点
题目大意 N个点的有向图中,定义“好点”为: 从该点v出发可以到达的所有点u,均有一条路径使得u可达v. 求出图中所有的“好点”,并按照顺序从小到大输出出来. 题目分析 图存在多个强连通分支,强连通分 ...
- Day04_数据类型占位符使用及进制转换
2013年10月09日 星期三 10时03分51秒 回顾: 1.变量 2.scanf标准函数 字符类型是一种数据类型 在C语言程序中字符类型用char表示 字符类型中包含了256个不同的字符,例如 ...
- 存储总量达20T的MySQL实例,如何完成迁移?
版权声明:本文由王亮原创文章,转载请注明出处: 文章原文链接:https://www.qcloud.com/community/article/122 来源:腾云阁 https://www.qclou ...
- windows环境下最简单的nginx + tomcat负载均衡配置示例
后端是两台tomcat服务器,我们简称为node1 和node2,访问地址分别是 http://192.168.1.2:8080 和 http://192.168.1.4:8080 前端使用nginx ...