题解 洛谷P5018【对称二叉树】(noip2018T4)
\(noip2018\) \(T4\)题解
其实呢,我是觉得这题比\(T3\)水到不知道哪里去了
毕竟我比较菜,不大会\(dp\)
好了开始讲正事
这题其实考察的其实就是选手对D(大)F(法)S(师)的掌握程度
考完试有人说这题是马拉车,吓死我了
首先,你把数据读入之后,先用一个大法师把以每个节点为根的子树的大小和权值都预处理出来,方便待会剪枝
然后,你对以每个节点为根的子树,都判断一下以下条件(这时刚才处理的东西就有用了)
① 左子树和右子树的节点数是否相等
② 左子树和右子树的权值是否相等
③ 以当前节点为根的子树大小是不是超过答案
第三个很重要,不加(洛谷数据)最后一个点会TLE
有一个显而易见的剪枝:因为答案至少是1,所以大小为1的子树就不用check了,不然浪费常数
然后就是暴力判了
递归下去,建立两个队列,保存当前处理到的左子树上和右子树上的节点,判左子树当前节点的左儿子和右子树当前节点的右儿子权值是否相等,右子树当前节点的左儿子和左子树当前节点的右儿子权值是否相等(注意对应)
还有判下对应的节点有没有一个是空的一个没空的情况
如果不相等就返回
相等的话就扔进队列(注意对应顺序!)
注意:上述处理一定要左右子树一起做,不能先处理一边,再处理另一边,不然会WA
到最后如果都可以的话就return true
附考场代码
不得不说,为了能过,我加了一堆卡常
3e6的输入规模应该还是要快读的吧
# include <bits/stdc++.h>
# define R register
const int MaxN = 1000010;
struct node//节点
{
int val;
int l, r;
};
node a[MaxN];
int f[MaxN], val[MaxN], ind[MaxN];//f[i]表示以i为根的子树大小,val表示以i为根的子树权值和,ind没啥用
inline void read(int &x)//快读
{
x = 0;
bool op = 1;
char ch = getchar();
while(ch > '9' || ch < '0')
{
if(ch == '-')
op = 0;
ch = getchar();
}
while(ch <= '9' && ch >= '0')
x = (x << 1) + (x << 3) + (ch - '0'), ch = getchar();
if(!op)
x = -x;
}
void dfs(int root)
{
if(root == -1)
return;
if(a[root].l == -1 && a[root].r == -1)
f[root] = 1, val[root] = a[root].val;
else
{
dfs(a[root].l);
dfs(a[root].r);
f[root] = f[a[root].l] + f[a[root].r] + 1;
val[root] = val[a[root].l] + val[a[root].r] + a[root].val;
}
}
inline int check(int x)
{
std::queue<int> l, r;
l.push(x), r.push(x);
while(!l.empty() || !r.empty())
{
if(l.empty() || r.empty())
return false;//一边空了,一边没空
R int lx = l.front(), rx = r.front();
l.pop(), r.pop();
if(a[lx].val != a[rx].val)
return false;
R int lson[3], rson[3];
lson[1] = a[lx].l, lson[2] = a[lx].r;//左子树当前节点的左儿子,左子树当前节点的右儿子
rson[1] = a[rx].l, rson[2] = a[rx].r;//右子树当前节点的左儿子,右子树当前节点的右儿子
if((lson[1] == -1 && rson[2] != -1) || (lson[1] != -1 && rson[2] == -1))
return false;//一边空了,一边没空
if((lson[2] == -1 && rson[1] != -1) || (lson[2] != -1 && rson[1] == -1))
return false;//一边空了,一边没空
if(lson[1] != -1)
l.push(lson[1]);
if(lson[2] != -1)
l.push(lson[2]);
if(rson[2] != -1)
r.push(rson[2]);
if(rson[1] != -1)
r.push(rson[1]);
//推进队列
}
return true;
}
int main()
{
// freopen("tree.in", "r", stdin);
// freopen("tree.out", "w", stdout);
R int n;
scanf("%d", &n);
for(R unsigned i = 1; i <= n; ++i)
read(a[i].val);
for(R unsigned i = 1; i <= n; ++i)
read(a[i].l), read(a[i].r), ++ind[a[i].l], ++ind[a[i].r];//处理入度
R unsigned root;
for(R unsigned i = 1; i <= n; ++i)
{
if(!ind[i])
{
root = i;
break;
}
}//找树根
dfs(root);//预处理
int ans = 1;
for(R unsigned i = 1; i <= n; ++i)//枚举子树
{
if(f[a[i].l] != f[a[i].r])
continue;//剪枝1
if(val[a[i].l] != val[a[i].r])
continue;//剪枝2
if(f[i] < ans || f[i] == 1)
continue;//剪枝3
if(check(i))
ans = f[i];//更新答案
}
printf("%d", ans);
fclose(stdin);
fclose(stdout);
return 0;
}
题解 洛谷P5018【对称二叉树】(noip2018T4)的更多相关文章
- 洛谷P5018 对称二叉树——hash
给一手链接 https://www.luogu.com.cn/problem/P5018 这道题其实就是用hash水过去的,我们维护两个hash 一个是先左子树后右子树的h1 一个是先右子树后左子树的 ...
- 洛谷P5018 对称二叉树
不多扯题目 直接题解= = 1.递归 由题目可以得知,子树既可以是根节点和叶节点组成,也可以是一个节点,题意中的对称二叉子树是必须由一个根节点一直到树的最底部所组成的树. 这样一来就简单了,我们很容易 ...
- NOIP2018普及T4暨洛谷P5018 对称二叉树题解
题目链接:https://www.luogu.org/problemnew/show/P5018 花絮:这道题真的比历年的t4都简单的多呀,而且本蒟蒻做得出t4做不出t3呜呜呜... 这道题可以是一只 ...
- 洛谷 P5018 对称二叉树(搜索)
嗯... 题目链接:https://www.luogu.org/problem/P5018 其实这道题直接搜索就可以搜满分: 首先递归把每个点作为根节点的儿子的数量初始化出来,然后看这个节点作为根节点 ...
- 【洛谷P5018 对称二叉树】
话说这图也太大了吧 这题十分的简单,我们可以用两个指针指向左右两个对称的东西,然后比较就行了 复杂度O(n*logn) #include<bits/stdc++.h> using name ...
- 洛谷 P5018 对称二叉树
题目传送门 解题思路: 先计算每个点的子树有多少节点,然后判断每个子树是不是对称的,更新答案. AC代码: #include<iostream> #include<cstdio> ...
- 2021.08.09 P5018 对称二叉树(树形结构)
2021.08.09 P5018 对称二叉树(树形结构) [P5018 NOIP2018 普及组] 对称二叉树 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 题意: 求一棵子树,关 ...
- 【洛谷P5018】对称二叉树
题目大意:定义对称二叉树为每个节点的左右子树交换后与原二叉树仍同构的二叉树,求给定的二叉树的最大对称二叉子树的大小. 代码如下 #include <bits/stdc++.h> using ...
- P5018 对称二叉树题解
题目内容链接: 那么根据题意,上图不是对称二叉树,只有节点7的子树是: 通俗来说,对称二叉树就是已一个节点x为根的子树有穿过x点的对称轴并且对称轴两边的对称点的大小也必须相等,那么这棵树就是对称二叉树 ...
随机推荐
- Foxmail7.2的账号密码的备份与恢复
1:备份: 1.0 找到Foxmail7.2的安装的位置,例如我的就是 D:\Program Files\Foxmail 7.2; 然后在路径下找到Storage文件夹然后备份里边的内容; 邮箱账号备 ...
- js:把字符串转为变量使用; js下将字符串当函数去执行的方法
1 把字符串当变量使用 通过计算 string 得到的值(如果有的话).该方法只接受原始字符串作为参数 demo: var type = "car"; var newStr = & ...
- Spring Boot 多个域名指向同一IP
一.需求:直接通过域名访问首页(同一应用下,多个首页,包括PC端.手机端首页) 方法:采用多个域名绑定同一IP下同一应用,不同域名对应不同产品(PC.手机端)的方法,在后台通过拦截器判断 reques ...
- 【转载】SpringBoot yml 配置
1. 在 spring boot 中,有两种配置文件,一种是application.properties,另一种是application.yml,两种都可以配置spring boot 项目中的一些变量 ...
- 笔记: ASP.NET Core视图组件
视图组件 asp.net core mvc 提供了部分视图的新替代品:视图组件. 视图组件与分布视图的主要区别在于视图组件与控制器不相关.可使用在独立于单个控制器的场景,如:菜单导航.侧边栏.分页栏等 ...
- 编译 SharpNav 遇到的问题和解决过程
https://github.com/Robmaister/SharpNav 是github上基于recastnavtigation的一个C#项目. github上并没有详细的编译过程. 首先把项目c ...
- java计算接口调用时间
方法一: LocalDateTime beginTime = LocalDateTime.now(); Long opetime = Duration.between(between,LocalDat ...
- python matplotlib动态绘图
matplotlib animation的官方文档: http://matplotlib.org/api/animation_api.html 接下来完成一个实时获取cpu数值,并绘图的功能. 1.动 ...
- 解决wpscan无法更新
如果wpscan无法更新的话 一般的原因都是源或者更新地址无法访问 下面解决 updatedb #先更新一下系统的索引 locate wpscan #定位到wpscan的目录 大概就是updater. ...
- CSS3弹性盒子(Flex Box)
CSS3弹性盒子(Flex Box) 一.容器的属性:flex-directionflex-wrapflex-flowjustify-contentalign-itemsalign-content 1 ...