BZOJ.4919.[Lydsy1706月赛]大根堆(线段树合并/启发式合并)
考虑树退化为链的情况,就是求一个最长(严格)上升子序列。
对于树,不同子树间是互不影响的。仿照序列上的LIS,对每个点x维护一个状态集合,即合并其子节点后的集合,然后用val[x]替换掉第一个大于它的数(有等于的就不换了)。
最后根节点状态集合的大小就是答案了。
关于替换数,可以先找到这个数的位置,如果有这个数就不用管了;没有的话插入进去,然后递归回去,找到一个靠右的位置删掉。
当然其实不用线段树合并这么麻烦,直接上multiset启发式合并就可以了。。
注意是合并了子树的状态,so叶节点size>1也是可能的!不要直接置为0!(应该-=1)
线段树合并比set启发式合并慢多了。。。
DFS可以用for一遍n~1取代。
另外有另一种线段树合并的方法,要区间修改、标记永久化,这里大体记一下具体步骤:
以下是口胡,建议忽略。
\(x\)从子树转移,就是\(f[x][j]=f[v1][j]+f[v2][j]+...\),对应位置相加;最后转移完对区间\(val_x\sim INF\)和\(f[x][val_x-1]+1\)取\(\max\)。
\(CQzhangyu\)的代码里\(tag\)是区间取\(\max\)的标记,\(sum\)是区间加的标记。\(sum\)和正常一样在访问节点前下传。\(tag\)为什么感觉不标记永久化一样下传也可以啊...当然也没必要下传。
合并的时候,假设现在是把\(y\)合并到\(x\)上(是对应位置做加法)。
如果\(lson[x]=0\),那说明\(lson[x]\)的最大值就是\(x\)的最大值\(tag[x]\),所以之前\(x\)上的给\(lson[x]\)的区间加标记\(sum[x]\)就不用管了。(\(x\)节点上的最大值是已经被\(sum\)更新后的)
所以如果\(lson[x]=0,\ lson[y]\neq0\),合并是对应位置相加,所以\(lson[x]\)的值应该变成\(lson[x]+lson[y]\)。
而现在\(lson[x]=0\),而\(tag[lson[x]]\)其实等于\(tag[x]\),所以对应位置相加后,现在的\(tag[lson[x]]\)就变成\(tag[x]+tag[lson[y]]\)。
\(lson[y]\)的区间加标记可能没下传,要保留,现在加上\(lson[x]\)的值就变成了\(tag[x]+sum[lson[y]]\)。
其它时候同理吧...
//64956kb 1288ms
#include <cstdio>
#include <cctype>
#include <cassert>
#include <algorithm>
//#define gc() getchar()
#define MAXIN 300000
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
const int N=2e5+5;
int n,A[N],cnt,ref[N],root[N],Enum,H[N],nxt[N],to[N];
char IN[MAXIN],*SS=IN,*TT=IN;
struct Segment_Tree
{
#define S N*19
#define lson son[x][0]
#define rson son[x][1]
int tot,sz[S],son[S][2];
bool flag;
#define Update(x) sz[x]=sz[lson]+sz[rson]
void Delete(int x,int l,int r)
{
if(l==r)
{
flag=0, --sz[x];//not sz[x]=0!
return;
}
if(sz[lson]) Delete(lson,l,l+r>>1);
else/*if(sz[rson])*/ Delete(rson,(l+r>>1)+1,r);
Update(x);
}
void Insert(int &x,int l,int r,int p)
{
if(!x) x=++tot;
if(l==r)
{
if(!sz[x]) flag=1, ++sz[x];
return;
}
int m=l+r>>1;
if(p<=m)
{
Insert(lson,l,m,p);
if(flag&&sz[rson]) Delete(rson,m+1,r);//新建的时候肯定删不了啊(没有右子树)
}
else Insert(rson,m+1,r,p);
Update(x);
}
int Merge(int x,int y)
{
if(!x||!y) return x^y;
lson=Merge(lson,son[y][0]), rson=Merge(rson,son[y][1]);
sz[x]+=sz[y], sz[y]=0; return x;
}
}T;
inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
inline void AddEdge(int u,int v){
/*if(u)*/ to[++Enum]=v, nxt[Enum]=H[u], H[u]=Enum;
}
int Find(int x)
{
int l=1, r=cnt, mid;
while(l<r)
if(ref[mid=l+r>>1]<x) l=mid+1;
else r=mid;
return l;
}
void DFS(int x)
{
for(int i=H[x]; i; i=nxt[i])
DFS(to[i]), root[x]=T.Merge(root[x],root[to[i]]);
T.flag=0, T.Insert(root[x],1,cnt,A[x]);
}
int main()
{
n=read();
for(int i=1; i<=n; ++i) ref[i]=A[i]=read(), AddEdge(read(),i);
std::sort(ref+1,ref+1+n), cnt=1;
for(int i=2; i<=n; ++i) if(ref[i]!=ref[i-1]) ref[++cnt]=ref[i];
for(int i=1; i<=n; ++i) A[i]=Find(A[i]);
DFS(1), printf("%d\n",T.sz[root[1]]);
return 0;
}
BZOJ.4919.[Lydsy1706月赛]大根堆(线段树合并/启发式合并)的更多相关文章
- bzoj 4919 [Lydsy1706月赛]大根堆 set启发式合并+LIS
4919: [Lydsy1706月赛]大根堆 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 599 Solved: 260[Submit][Stat ...
- BZOJ4919[Lydsy1706月赛]大根堆-------------线段树进阶
是不是每做道线段树进阶都要写个题解..根本不会写 Description 给定一棵n个节点的有根树,编号依次为1到n,其中1号点为根节点.每个点有一个权值v_i. 你需要将这棵树转化成一个大根堆.确切 ...
- BZOJ 4919: [Lydsy1706月赛]大根堆 启发式合并
我不会告诉你这是线段树合并的好题的... 好吧我们可以搞一个multiset在dfs时求出LIS(自带二分+排序)进行启发式合并,轻松加愉悦... #include<cstdio> #in ...
- BZOJ 4919: [Lydsy1706月赛]大根堆
F[x][i]表示x的子树中取的数字<=i的最大值,线段树合并优化DP 写得很难看,并不知道好看的写法 #include<cstdio> #include<algorithm& ...
- BZOJ 4919: [Lydsy1706月赛]大根堆 set启发式合并
这个和 bzoj 5469 几乎是同一道题,但是这里给出另一种做法. 你发现你要求的是一个树上 LIS,而序列上的 LIS 有一个特别神奇的 $O(n\log n) $ 做法. 就是维护一个单调递增的 ...
- BZOJ 4919 [Lydsy1706月赛]大根堆 (SRM08 T3)
[题解] 求一个序列的LIS有一个二分做法是这样的:f[i]表示长度为i的上升序列中最后一个数最小可以是多少,每次二分大于等于当前数字x的f[j],把f[j]修改为x:如果找不到这样的f[j],那就把 ...
- 【BZOJ4919】[Lydsy六月月赛]大根堆 线段树合并
[BZOJ4919][Lydsy六月月赛]大根堆 Description 给定一棵n个节点的有根树,编号依次为1到n,其中1号点为根节点.每个点有一个权值v_i. 你需要将这棵树转化成一个大根堆.确切 ...
- [Lydsy1706月赛]大根堆
4919: [Lydsy1706月赛]大根堆 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 358 Solved: 150[Submit][Stat ...
- bzoj4919 [Lydsy1706月赛]大根堆
Description 给定一棵n个节点的有根树,编号依次为1到n,其中1号点为根节点.每个点有一个权值v_i. 你需要将这棵树转化成一个大根堆.确切地说,你需要选择尽可能多的节点,满足大根堆的性质: ...
随机推荐
- Web 前端开发规范文档
通用规范: TAB键用两个空格代替(WINDOWS下TAB键占四个空格,LINUX下TAB键占八个空格). CSS样式属性或者JAVASCRIPT代码后加“;”方便压缩工具“断句”. 文件内容编码均统 ...
- Mysql注入root权限直接写一句话马
首先我们的找到一个有注入的站:这里我用自己搭建的环境表示:大家不要乱来 http://localhost/pentest/sql/sql_injection_get.php?id=1 发现是root权 ...
- Java IO,硬骨头也能变软
开胃菜 先看一张网上流传的http://java.io包的类结构图: 当你看到这幅图的时候,我相信,你跟我一样内心是崩溃的. 有些人不怕枯燥,不怕寂寞,硬着头皮看源码,但是,能坚持下去全部看完的又有几 ...
- python随笔(三)
在对字符串的操作中,s[::-1]表示将字符串逆序输出. 字符串本身不能改变(管理者而非所有者) 列表的内容是可以改变的,且列表的内容可以不仅仅是字符串.对于一个列表,注意b=a和b=a[:]的区别. ...
- 洛谷P1342请柬
传送门啦 核心思想:两遍最短路. 1号点去各地的时间直接套最短路模板,各地到1号点时间用逆向思维,视为求1号点沿反边到各地的时间即可. #include <iostream> #inclu ...
- 洛谷P1301 魔鬼之城
传送门啦 一道广度优先搜索的题目. 结构体含义: struct node{ int x,y,dir;//坐标,方向 int step;//当前步数 }; 方向的标号受上面定义的 $ dx[ ] , d ...
- java IO流知识点总结
I/O类库中使用“流”这个抽象概念.Java对设备中数据的操作是通过流的方式.表示任何有能力产出数据的数据源对象,或者是有能力接受数据的接收端对象.“流”屏蔽了实际的I/O设备中处理数据的细节.IO流 ...
- hiho 1227 找到一个恰好包含n个点的圆 (2015北京网赛 A题)
平面上有m个点,要从这m个点当中找出n个点,使得包含这n个点的圆的半径(圆心为n个点当中的某一点且半径为整数)最小,同时保证圆周上没有点. n > m 时要输出-1 样例输入43 2 0 0 1 ...
- JavaScript工程师都应懂的33个概念
最近福利发的有点多啊,各种硬干货,小伙伴们是不是觉得很爽啊.Github真的蕴含着各种各样的宝藏,难怪各个大厂也都纷纷贡献自己的代码到Github上. 所以各种干货还是会源源不断的po给大家,觉得有帮 ...
- C++ 内存分配(new,operator new)详解
参考:C++ 内存分配(new,operator new)详解 如何限制对象只能建立在堆上或者栈上 new运算符和operator new() new:指我们在C++里通常用到的运算符,比如A* a ...