【百度之星2014~复赛)解题报告】The Query on the Tree
声明
笔者最近意外的发现 笔者的个人网站http://tiankonguse.com/ 的很多文章被其它网站转载,但是转载时未声明文章来源或参考自 http://tiankonguse.com/ 网站,因此,笔者添加此条声明。
郑重声明:这篇记录《【百度之星2014~复赛)解题报告】The Query on the Tree》转载自 http://tiankonguse.com/ 的这条记录:http://tiankonguse.com/record/record.php?id=673
前言
这几天把毕业答辩的事弄完了,于是买票出来玩,结果周六是百度之星的复赛,于是我就没有办法来做比赛了,不过看了看题,目测可以过我两三道题.
今天已经是比赛的第二天了,我还一直没有时间来A掉这些题,今晚抽空先把最简单的线段树那道题A了再说.
正文
题意
题目说的很清楚了,自己看吧.
有一棵树,树的每个点有点权,每次有三种操作:
1. Query x 表示查询以x为根的子树的权值和。
2. Change x y 表示把x点的权值改为y(0<=y<=100)。
3. Root x 表示把x变为根。
分析
这道题的数据起始很弱的.
我最初的想法就可以把这道题过掉.
最初的想法
首先对这个树按1为根dfs根优先编号,这个应该没有什么疑问.
编号的好处是一个子树变为了一个连续的区间.
编号的时候保存一下这个子树的编号区间,保存在子树的根上.
编号的时候顺便计算一下子树的权值和.
编号的时候记录一下一个节点的父节点.
修改操作
先说说修改操作,修改某个节点时,就算出这个节点应该增加多少,然后从这个节点开始更新,一直更新到根1.
平均复杂度 O( log(n) )
最坏复杂度 O( n )
设置根
这里我们需要一个变量来表示目前的根是那个节点,比如使用root变量,默认值是1.
设置根只需要把根变量更新一下即可.
平均复杂度 O( 1 )
最坏复杂度 O( 1 )
查询操作
查询的时候分三种情况:
1.查询的节点是目前的根
这个时候答案显然是整个树的权值和,返回 根1的权值和即可.
2.目前的根不是查询的节点的某个子孙(即根不在查询的子树里面)
这个时候,答案和根是1的情况相同,及直接返回查询节点的权值和即可.
怎么判断根是不是查询节点的子孙呢?
平常的方法是用 LCA 查询,这里我直接使用子树区间来判断即可.
3.目前的根是查询节点的某个子孙.
这个时候,我们想象一下,我们拿起根,查询节点的子孙有那些呢?
即那些会在查询节点的下面呢?
假设查询节点是 x, x的一个儿子是y, 根是y的一个子孙(也可能是y).
这个时候,我们拿起根,x 应该变成 y 的儿子了吧.
这时树的权值应该是 x 原先的权值和 - y 节点的权值和 + 不在x子树区间的全职和.
然后,我们可以发现 x 原先的权值和 + 不在x子树区间的权值和 = 整个树的权值和.
故最终答案是 整个树的权值和 - y节点的权值和.
问题:怎么找到y节点.
有两个方法:
1.枚举x的儿子来判断
2.从根不断的找父亲来判断.
由于题意没有说最多儿子有多少个,所以第一个方法最坏情况下为 O( n ) (很多儿子)
对应的,第二个方法最坏情况下也是 O( n ) (树退化为链表).
不过我们不用管最坏情况,先这样实现了再说.
综合操作复杂度:log(n)
线段数优化
首先对于修改操作,线段树优化后可以使最坏情况达到 O( log( n ) ).
对于查询操作,由于需要知道 x 的那个儿子 y, 这个我目前没有想到 O( log( n ) ) 的方法.
学弟说那只能使用二分了.
但是怎么二分呢?
发现二分不了,不过可以使用随机算法来优化找儿子的效率.
起初我们是遍历x的所有儿子,这里我们随机挑一个儿子来寻找.这也算是一个比较好的优化方法吧.
代码
暴力版代码 https://github.com/tiankonguse/ACM/blob/master/astar/2014/3/2.2.cpp(比较简洁)
线段树优化版代码 https://github.com/tiankonguse/ACM/blob/master/astar/2014/3/2.cpp
对于上面说的几个方法我只实现了两个,其他的都很简单,有兴趣的朋友可以尝试一下.
参考
http://blog.csdn.net/hongrock/article/details/27839237(这个参考主要用于确认暴力不会超时,如果精心构造数据,这个方法会超时的)
【百度之星2014~复赛)解题报告】The Query on the Tree的更多相关文章
- 【百度之星2014~复赛 解题报告~正解】The Query on the Tree
声明 笔者最近意外的发现 笔者的个人网站http://tiankonguse.com/ 的很多文章被其它网站转载,但是转载时未声明文章来源或参考自 http://tiankonguse.com/ 网站 ...
- 百度之星2014复赛 - 1002 - The Query on the Tree
先上题目: The Query on the Tree Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (J ...
- 百度之星2014复赛 - 1001 - Find Numbers
先上题目: Find Numbers Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Othe ...
- 【百度之星2014~初赛(第二轮)解题报告】Chess
声明 笔者近期意外的发现 笔者的个人站点http://tiankonguse.com/ 的非常多文章被其他站点转载.可是转载时未声明文章来源或參考自 http://tiankonguse.com/ 站 ...
- 【百度之星2014~初赛(第二轮)解题报告】JZP Set
声明 笔者近期意外的发现 笔者的个人站点http://tiankonguse.com/ 的非常多文章被其他站点转载,可是转载时未声明文章来源或參考自 http://tiankonguse.com/ 站 ...
- [百度之星2014资格赛] Disk Schedule 报告
Disk Schedule Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) To ...
- NOIP2016普及组复赛解题报告
提高组萌新,DAY1DAY2加起来骗分不到300,写写普及组的题目聊以自慰. (附:洛谷题目链接 T1:https://www.luogu.org/problem/show?pid=1909 T2:h ...
- 百度之星2014初赛 - 1002 - Grids
先上题目: Grids Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65535/65535 K (Java/Others)Tota ...
- 百度之星2014资格赛 1003 - Xor Sum
先上代码: Xor Sum Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 132768/132768 K (Java/Others)T ...
随机推荐
- [LeetCode 题解]: palindromes
Determine whether an integer is a palindrome. Do this without extra space. Some hints: Could negativ ...
- c# mvc 第三方定时 FluentScheduler
开头无关内容,可略过: 最近工作和生活都忙的不可开交了. 工作和生活上都不太顺利,5月底出了车祸回家养伤.6月忙着今年的大事. 给自己鼓鼓励吧 最近重拾nodejs和python,也给自己动力,继续学 ...
- ML.NET Cookbook --- 1.如何从文本文件中加载数据?
使用ML.NET中的TextLoader扩展方法从文本文件中加载数据.你需要知道在文本文件中数据列在那里,它们的类型是什么,在文本文件中什么位置可以找到它们. 请注意:对于ML.NET只读取文件的某些 ...
- [原创] 思维导图笔记(二):SQL
为了补一补知识体系上的缺漏,最近找了看完了Ben Forta写的<Sams Teach Yourself SQL in 10 Minutes>,翻译过来叫<SQL必知必会>.从 ...
- HAOI2010 最长公共子序列
题目链接:戳我 30分暴力....暴力提取子序列即可qwqwq #include<iostream> #include<cstdio> #include<algorith ...
- libz.dylib框架怎么导入
1.General下 2.点击+号在弹出的对话框选择addother 3.在弹出的对话框中输入"cmd"+"shift"+"g" 输入/us ...
- linux下的常用指令
1,在vim中查找字符段 :1?字段名,此方式可以从开始向下查询字段了. :?字段名 ,查询字都段: 2,修改某个文件夹用户和组 修改文件所属用户:chown [-R] 用户 文件或目录 如:chow ...
- Linux系统查找清理磁盘大文件方法
本文主要介绍Linux系统磁盘使用空间不足时,如何查找大文件并进行清理的方法. 下午使用df-h检查一台服务器磁盘使用空间,发现磁盘已经使用了100%,其中/dev/mapper/vg_iavp-lv ...
- 理解webpack中的publicPath
outPut中的publicPath 默认值: 空字符串. publicPath是非常有必要配置的,他是项目中引入静态资源(js.css)时的基础路径. 例如: outPut.publicPath = ...
- Ubuntu14.04安装libusb
https://www.cnblogs.com/ettie999/p/8142973.html libuvc是一个跨平台的USB视频设备库,建立在libusb之上. 它能够对导出标准USB视频类(UV ...