题目地址:http://ac.jobdu.com/problem.php?pid=1541

题目描述:

旋转是二叉树的基本操作,我们可以对任意一个存在父亲节点的子节点进行旋转,包括如下几种形式(设被旋转节点为x,其父亲节点为p):

1.左旋

旋转前,x是p的右儿子。

x的左儿子(若存在)变为p的右儿子,p变为x的左儿子。如下图

2.右旋

旋转前,x是p的左儿子。

x的右儿子(若存在)变为p的左儿子,p变为x的右儿子。如下图



综上,我们可以通过检查选择前x是p的左儿子还是右儿子来判断该次旋转是左旋还是右旋。

给定一颗n个节点的二叉树,其节点由1至n编号,并给定一系列操作,如下:

1.rotate x,对编号x的节点进行旋转,若x为根节点,则不进行任何操作。

2.parent x,输出编号x的父亲节点编号,若x为根节点输出-1。

3.size x,输出以x为根节点的子树的节点个数。

输入:

输入包含多组测试用例。

每组测试用例开头为一个整数n(1<=n<=1000),代表二叉树的节点个数。

接下去n行描述,二叉树原始的状态,第i行为两个整数x,y,代表i号节点的左儿子节点为x号节点,右儿子节点为y号节点,若x或y为-1,则表示相应儿子节点不存在。编号的范围为1到n。

接下去一行为一个整数t(1<=t<=50000),代表操作的个数。

最后t行,每行代表一个对二叉树的操作,描述如上所示。

输出:

对于每组测试用例,输出操作parent x和size x查询的数据。

样例输入:
5
2 3
-1 -1
4 5
-1 -1
-1 -1
5
size 1
rotate 5
size 5
parent 3
parent 4
样例输出:
5
3
5
3

#include <stdio.h>
#include <string.h> typedef struct node{
int parent;
int left;
int right;
}Node; Node tree[1001];
int has_parent[1001];
int size[1001];
int n;
int root; int Compute(int node){
return (node == -1) ? 0 : size[node];
} int Size(int node){
if (node == -1)
return 0;
if (tree[node].left != -1)
size[tree[node].left] = Size(tree[node].left);
if (tree[node].right != -1)
size[tree[node].right] = Size(tree[node].right);
return size[node] = Compute(tree[node].left) + Compute(tree[node].right) + 1;
} void Rotate(int node){
int parent, grandpar;
int left, right;
if (node != root){
parent = tree[node].parent;
grandpar = tree[parent].parent;
tree[node].parent = grandpar;
if (grandpar != -1){
if (tree[grandpar].right == parent)
tree[grandpar].right = node;
else
tree[grandpar].left = node;
}
if (parent == root)
root = node;
tree[parent].parent = node;
if (tree[parent].right == node){//node是其父节点的右孩子,左旋
left = tree[node].left;
tree[parent].right = left;
tree[node].left = parent;
if (left != -1){
tree[left].parent = parent;
}
}
else{//node是其父节点的左孩子,右旋
right = tree[node].right;
tree[parent].left = right;
tree[node].right = parent;
if (right != -1){
tree[right].parent = parent;
}
}
size[node] = size[parent];
size[parent] = Compute(tree[parent].left) + Compute(tree[parent].right) + 1;
}
} int main(void) {
int i;
int t;
char ope[10];
int id; while (scanf("%d", &n) != EOF){
for (i = 1; i <= n; ++i){
memset(has_parent, 0, sizeof(has_parent));
memset(size, 0, sizeof(size));
scanf("%d%d", &tree[i].left, &tree[i].right);
if (tree[i].left != -1){
tree[tree[i].left].parent = i;
has_parent[tree[i].left] = 1;
}
if (tree[i].right != -1){
tree[tree[i].right].parent = i;
has_parent[tree[i].right] = 1;
}
}
for (i = 1; i <= n; ++i)
if (has_parent[i] != 1){
tree[i].parent = -1;
root = i;
break;
}
for (i = 1; i <= n; ++i)
Size(i);
scanf("%d", &t);
while (t-- != 0){
scanf("%s%d", ope, &id);
if (ope[0] == 'r')
Rotate(id);
else if (ope[0] == 'p')
printf("%d\n", tree[id].parent);
else
printf("%d\n", size[id]);
}
} return 0;
}

九度OJ 1541 二叉树【数据结构】的更多相关文章

  1. 九度oj 1541 二叉树

    原题链接:http://ac.jobdu.com/problem.php?pid=1541 简答题如下: #include<algorithm> #include<iostream& ...

  2. 九度oj 1184 二叉树遍历

    原题链接:http://ac.jobdu.com/problem.php?pid=1184 简单的二叉树重建,遍历. 如下: #include<cstdio> #include<cs ...

  3. [九度OJ]1078.二叉树的遍历(重建)

    原题链接:http://ac.jobdu.com/problem.php?pid=1078 题目描述: 二叉树的前序.中序.后序遍历的定义:前序遍历:对任一子树,先访问跟,然后遍历其左子树,最后遍历其 ...

  4. [九度OJ]1113.二叉树(求完全二叉树任意结点所在子树的结点数)

    原题链接:http://ac.jobdu.com/problem.php?pid=1113 题目描述: 如上所示,由正整数1,2,3……组成了一颗特殊二叉树.我们已知这个二叉树的最后一个结点是n.现在 ...

  5. 九度OJ 1113 二叉树

    题目地址:http://ac.jobdu.com/problem.php?pid=1113 题目描述: 如上所示,由正整数1,2,3……组成了一颗特殊二叉树.我们已知这个二叉树的最后一个结点是n.现在 ...

  6. 九度OJ 1078 二叉树遍历

    题目地址:http://ac.jobdu.com/problem.php?pid=1078 题目描述: 二叉树的前序.中序.后序遍历的定义: 前序遍历:对任一子树,先访问跟,然后遍历其左子树,最后遍历 ...

  7. 九度oj 1521 二叉树的镜像

    原题链接:http://ac.jobdu.com/problem.php?pid=1521 水题,如下.. #include<algorithm> #include<iostream ...

  8. 【九度OJ】题目1113:二叉树 解题报告

    [九度OJ]题目1113:二叉树 解题报告 标签(空格分隔): 九度OJ http://ac.jobdu.com/problem.php?pid=1113 题目描述: 如上所示,由正整数1,2,3-- ...

  9. 【九度OJ】题目1078:二叉树遍历 解题报告

    [九度OJ]题目1078:二叉树遍历 解题报告 标签(空格分隔): 九度OJ http://ac.jobdu.com/problem.php?pid=1078 题目描述: 二叉树的前序.中序.后序遍历 ...

随机推荐

  1. 使用IOCTL代码实现LCD背光调节

    国内这种代码找不到.于是參考了相关代码后完好例如以下代码,且实现方式通过IOCTL代码实现LCD背光调节的功能. 适合场合为平板电脑或者笔记本.主要还是要靠BIOS支持与否. 编译环境使用:Dev-c ...

  2. ios开发——面试篇(一)

    面试篇之内存管理与多线程 简述OC中内存管理机制.­­­­­与retain配对使用的方法是dealloc还是release,为什么?需要与alloc配对使用的方法是dealloc还是release,为 ...

  3. stl 容器

    10.1.2.2容器的分类 序列式容器(Sequence containers) 每个元素都有固定位置--取决于插入时机和地点,和元素值无关. vector.deque.list  关联式容器(Ass ...

  4. Python学习 之 运算符&表达式

    1.Python运算符包括:赋值运算符.算术运算符.关系运算符.逻辑运算符. 表达式是将不同的数据(包括变量.函数)用运算符号按一定规则连接起来的一种式子. 2.赋值运算符:=.+=.-=.*=./= ...

  5. 通用PE u盘装Ghost Win7系统

    http://www.tongyongpe.com/win7ghost.html 导读 通用pe工具箱是现在最老牌的的U盘装系统和维护电脑的专用工具之一,一键式制作.操作简单便捷,几乎100%支持所有 ...

  6. objc_msgSend消息传递学习笔记 – 消息转发

    该文是 objc_msgSend消息传递学习笔记 – 对象方法消息传递流程 的基础上继续探究源码,请先阅读上文. 消息转发机制(message forwarding) Objective-C 在调用对 ...

  7. 大文件读取方法(C#)

    之前都是用StreamReader.ReadLine方法逐行读取文件,自从.NET4有了File.ReadLines这一利器,就再也不用为大文件发愁了. File.ReadLines在整个文件读取到内 ...

  8. css笔记14:css文件之间可以相互引用

    css文件之间相互引用是通过@import指令完成的 格式: @import  url("被引用的css文件"); 顺便说一句,如果希望在html或者php文件中引用某个xxx.c ...

  9. T-SQL利用Row_Number函数实现分页

    SQL: CREATE PROCEDURE PagingViewTest ( @currentPageIndex INT, --页序号 @pageSize INT, --页大小 @pageCount ...

  10. EF中"实体类型 XXXXX 不是当前上下文的模型的一部分。" 原因

    用T4模版生成的数据库映射文件.cs对应不上数据库的字段而引起的错误(我是修改了cs里面的属性),只要根据数据库字段,修改回来就OK.问题就解决了.