【问题描述】
众所周知,zzh 和 heyi 是一对竹马竹马,他们从小一起学 C++,最后都成了著名的神犇。而时间回溯到他们童年,这天 zzh 邀请 heyi 来参加 zzh 举行的男性家庭聚会。
而 zzh 则在苦恼该邀请哪些人。
我们按照一定的顺序给家庭成员编号,为了表示尊敬,我们给辈分最大的人编号为
1。每个人都有一定的身价,为了避免尴尬,我们希望邀请的所有家庭成员中每个人的身价都必须要小于他的直系长辈,此时 zzh 想知道最多能邀请多少人。
注意:直系长辈只包含父亲祖父等,叔叔不算。
【输入格式】
输入文件名为 chikuma.in。
输入文件第一行为一个正整数 n。
接下来 n 行,每行两个非负整数vi,pi,表示 i 号成员的身价与父亲。
【输出格式】
输出文件名为 chikuma.out。
输出文件包含 1 行,一个正整数,表示最多能邀请的人数。
【输入输出样例】
chikuma.in
chikuma.out
6
3 0
1 1
2 1
3 1
4 1
5 1
5
【输入输出样例说明】
2~6 号家庭成员的直系长辈为 1 号家庭成员。
如果 zzh 不邀请 1 号家庭成员,那么 2~6 号家庭成员都能参加,能够邀请 5 个人。
如果 zzh 邀请 1 号家庭成员,那么由于 3 号、4 号和 5 号家庭成员身价不低于 1 号
家庭成员,将不能被邀请,因此能够邀请 3 个人。
因此,最多能邀请 5 个人。
【数据范围】
对于 10%的数据,n≤10。
对于 40%的数据,n≤1000。
另有 10%的数据,保证不存在有叔叔的情况
对于 100%的数据,n≤200000,0≤vi≤1e9,
1≤pi<i,p1=0。
【题解】
对于40分做法:设f[i][j]表示以i为根的子树中选出的人最大身价的人为j的时候最多选出几个人,转移很简单。
对于100分做法:

线段树合并

线段树中叶节点表示最大为该身价时可选人数,非叶节点维护最大值,答案即为shu[root[1]].mx。

插入只需将线段树从根到表示v[u]的节点路径上,mx全部取max ((1-v[u]-1)中最大值+1)即可。

合并时右儿子对左边的所有的max取max。

即为区间取最大值,单点修改,区间查询。

维护该子树中身价最多为i时最多可选人数。

代码如下:

#include<bits/stdc++.h>
using namespace std;
inline int read(){
int x=,f=;char c=getchar();
while(!isdigit(c)){if(c=='-')f=-;c=getchar();}
while(isdigit(c)){x=(x<<)+(x<<)+(c^);c=getchar();}
return f==?x:-x;
}
int cn;
const int N=2e5+;
struct segment_tree{
#define lc ch[p][0]
#define rc ch[p][1]
int t[N<<],ch[N<<][],tot,lazy[N<<];
void pushup(int p)
{
t[p]=max(t[lc],t[rc]);
}
void pushdown(int p)
{
if(lc)
{
t[lc]+=lazy[p];
lazy[lc]+=lazy[p];
}
if(rc)
{
t[rc]+=lazy[p];
lazy[rc]+=lazy[p];
}
lazy[p]=;
}
void merge(int &p,int pre,int l,int r,int lx,int ly){
if(!p && !pre) return;
if(!pre)
{
lazy[p]+=ly;
t[p]+=ly;
return;
}
if(!p)
{
lazy[pre]+=lx;
t[pre]+=lx;
p=pre;
return;
}
if(l==r)
{
t[p]+=max(ly,t[pre]);
return;
}
int mid=(l+r)>>;
pushdown(p);
pushdown(pre);
merge(rc,ch[pre][],mid+,r,max(lx,t[lc]),max(ly,t[ch[pre][]]));
merge(lc,ch[pre][],l,mid,lx,ly);
pushup(p);
}
void insert(int &p,int l,int r,int x,int v){
if(!p)p=++tot;
t[p]=max(t[p],v);
if(l==r)return;
int mid=l+r>>;
if(x<=mid) insert(lc,l,mid,x,v);
else insert(rc,mid+,r,x,v);
}
int query(int p,int l,int r,int ql,int qr){
if(!p)return ;
if(ql<=l&&r<=qr)return t[p];
int mid=l+r>>,ret=;
pushdown(p);
if(ql<=mid)ret=max(ret,query(lc,l,mid,ql,qr));
if(mid<qr)ret=max(ret,query(rc,mid+,r,ql,qr));
return ret;
}
}seg;
struct edge{
int v,nxt;
}e[N];
int first[N],cnt=;
inline void add(int u,int v){
e[++cnt].v=v;e[cnt].nxt=first[u];first[u]=cnt;
}
int n,a[N],b[N],rt[N];
inline void dfs(int x){
for(int i=first[x],v,v1,v2;i;i=e[i].nxt){
v=e[i].v;
dfs(v);
v1=v2=;
seg.merge(rt[x],rt[v],,cn,v1,v2);
}
seg.insert(rt[x],,cn,a[x],((a[x]==)?:(seg.query(rt[x],,cn,,a[x]-)+)));
}
int main(){
freopen("chikuma.in","r",stdin);
freopen("chikuma.out","w",stdout);
long long size = << ;
char *p = (char*)malloc(size) + size;
__asm__("movl %0, %%esp\n" :: "r"(p));
n=read();
for(int i=,x;i<=n;i++){
b[i]=a[i]=read();x=read();
if(x) add(x,i);
}
sort(b+,b+n+);
cn=unique(b+,b+n+)-b-;
for(int i=;i<=n;i++)
a[i]=lower_bound(b+,b+cn+,a[i])-b;
dfs();
cout<<seg.t[rt[]];
fclose(stdin);
fclose(stdout);
return ;
}
/*
6
3 0 1 1 2 1 3 1 4 1 5 1
*/

竹马竹马chikuma的更多相关文章

  1. [Swust OJ 772]--Friend(并查集+map的运用)

    题目链接:http://acm.swust.edu.cn/problem/772/ Time limit(ms): 1000 Memory limit(kb): 65535    Descriptio ...

  2. python --常用内置模块01

    1.简单了解模块         模块就是我们把装有特定功能的代码进行归类的解构,从代码编写的单位来看我们的程序 从小到大的顺序:一条代码< 语句块<代码块(函数,类) < 模块 我 ...

  3. qwq。。胡诌qwq

    1,十里寒塘,红楼灯火阑珊,晓风残月,思念醒了一半. 晨钟催落月,淑气催黄鸟,而花千树而你还未还,三月烟花,只剩远影孤帆,珠箔飘灯,大雁独自来返,柳絮铺地,桃花落了菀晚, 琴声弹起,雨落长安,长夜漫漫 ...

  4. 佛山6397.7539(薇)xiaojie:佛山哪里有xiaomei

    佛山哪里有小姐服务大保健[微信:6397.7539倩儿小妹[佛山叫小姐服务√o服务微信:6397.7539倩儿小妹[佛山叫小姐服务][十微信:6397.7539倩儿小妹][佛山叫小姐包夜服务][十微信 ...

  5. 宜宾1178.9873(薇)xiaojie:宜宾哪里有xiaomei

    宜宾哪里有小姐服务大保健[微信:1178.9873倩儿小妹[宜宾叫小姐服务√o服务微信:1178.9873倩儿小妹[宜宾叫小姐服务][十微信:1178.9873倩儿小妹][宜宾叫小姐包夜服务][十微信 ...

随机推荐

  1. H3C 配置帧中继交换

  2. html(二)登陆页面

    今天开始正常上课学习HTML+CSS+JSP  嗯 前两个没讲直接上手! 老师也是很认同我们的呢~ 这是第一个案例 做一个登陆页面,并利用post提交表单 传值到另一个界面接收值. 1.设置值: &l ...

  3. H3C 显示RIP当前运行状态及配置信息

  4. [转]C#操作Memcached帮助类

    在VS中安装Memcached,直接在NuGet下搜索Memcached,选择第一个进行安装: 服务端资源下载地址:https://pan.baidu.com/s/1gf3tupl 接下来开始写程序, ...

  5. java Class中得到构造方法Constructor、方法Method、字段Field

    常用方法: Constructor类用于描述类中的构造方法: Constructor<T> getConstructor(Class<?>... parameterTypes) ...

  6. linux 在 scull 中使用旗标

    旗标机制给予 scull 一个工具, 可以在存取 scull_dev 数据结构时用来避免竞争情况. 但是正确使用这个工具是我们的责任. 正确使用加锁原语的关键是严密地指定要保护哪个 资源并且确认每个对 ...

  7. 【u105】路径计数2

    Time Limit: 1 second Memory Limit: 128 MB [问题描述] 一个N×N的网格,你一开始在(1, 1),即左上角.每次只能移动到下方相邻的格子或者右方相邻的格子,问 ...

  8. 深度学习——GAN

    整理自: https://blog.csdn.net/woaidapaopao/article/details/77806273?locationnum=9&fps=1 思想 表达式 实际计算 ...

  9. javascript中的深拷贝与浅拷贝

    javascript中的深拷贝与浅拷贝 基础概念 在了解深拷贝与浅拷贝的时候需要先了解一些基础知识 核心知识点之 堆与栈 栈(stack)为自动分配的内存空间,它由系统自动释放: 堆(heap)则是动 ...

  10. Python3装饰器的使用

    装饰器 简易装饰器模板 def wrapper(func): def inner(*args,**kwargs): print('主代码前添加的功能') ret=func(*args,**kwargs ...