Ants on tree

时间:2022-12-22 19:44:28

Description

  

  从前有一个策略游戏, 叫做 蚂蚁上树

  游戏中有一棵 nn 个节点, 以 11 为根的有根树

  初始始每个节点都为空, 游戏系统会进行两种操作 :

  1 x , 表示往 xx 节点放入一只睡眠状态中的蚂蚁

  2 x , 表示从 xx 节点取出一只睡眠状态中的蚂蚁

  (对于操作2, 保证取出前该点至少有一只蚂蚁)

  每次操作后, 玩家要进行一轮游戏 :

  游戏有无穷的时间, 每一时刻, 系统会 依次执行 下述五个操作

    1) 让玩家选择 任意多只(可以为 0 只) 睡眠状态中的蚂蚁

    2) 所有亢奋状态的蚂蚁朝根结点方向移动一步

    3) 若某一时刻 ≥2≥2只 亢奋状态 的蚂蚁处在同一节点, 游戏失败

    4) 到达根节点的蚂蚁进入睡眠状态.

    5) 当前时刻被玩家选择的蚂蚁进入亢奋状态

    6) 若所有蚂蚁都在根节点, 游戏结束

  游戏不允许失败, 玩家的游戏目的是 : 使游戏结束时, 最后一只到达根节点的蚂蚁到达时间最早.

  每轮游戏后, 系统会自动将树恢复成玩家该轮游戏前的局面, 然后进行下一次取/放蚂蚁的操作.

  

Input

  

  第一行两个数 n,mn,m 表示树的点数和操作数

  第 2−n2−n 行, 第 ii 行一个数 fifi 表示 ii 节点的父亲

  接下来 mm 行, 每行两个数表示系统的操作

  若为 1 x , 表示往 xx 节点放入一只睡眠状态中的蚂蚁

  若为 2 x , 表示从 xx 节点取出一只睡眠状态中的蚂蚁

  

Output

  

  输出 mm 行, 表示每轮游戏在最优策略下

  最后一只到达根节点的蚂蚁到达的最早时间

  (特别的, 如果所有蚂蚁都在根节点, 或者没有蚂蚁, 输出 0)

  

Sample Input

  

4 5
1
2
2
1 1
1 3
1 4
1 2
2 3

  

Sample Output

  

0
3
4
4
3
``` 
  
## HINT
  
  对于样例输出第四行的解释 :
  第一时刻触碰位于 2, 3 的那只蚂蚁, 他们进入亢奋状态但没有移动
  第二时刻触碰位于 4 的那只蚂蚁, 然后位于 2, 3 的蚂蚁分别爬到 1, 2, 然后爬到 1 的蚂蚁进入睡眠状态, 之后 4 进入亢奋状态.
  第三时刻不触碰蚂蚁, 当前位于 2, 4 的蚂蚁分别爬到 1, 2, 爬到 1 的这只蚂蚁进入睡眠状态
  第四时刻不触碰蚂蚁, 当前位于 2 的蚂蚁爬到 1 并进入睡眠状态, 然后游戏结束
  数据范围 :
  对于 30%的数据, $n,m≤3000$
  对于另外 30% 的数据, $n≤5000$
  对于另外 5% 的数据, 树的最大深度为 2
  对于另外 10% 的数据, 数据的生成方式如下 $fi=rand()%(i−1)+1$
  对于 100% 的数据 :
  $2≤n≤10^5$
  $1≤m≤10^5$
  $1≤fi<i,i=2..n$
  $1≤x≤n$
  
  
  
# Solution
  
  这题看起来很复杂,考场上我想了个反推的模拟,还写挂了,看起来并不对。
  
  考虑每个蚂蚁往上走的过程,是一激活就停不下来的。那么如果两只蚂蚁到达根节点的时间相同,那么它们必定在某一处会相撞。
  
  那么问题等价于为每一只蚂蚁挑一个至少为其深度的到达时间,使得每只蚂蚁的到达时间唯一,且最大时间最小。
  
​   我们应该按深度从小到大考虑每一个深度$i$的$sum_i$只蚂蚁,给同一深度的$sum_i$只蚂蚁挑连续一段的到达时间$[i,i+sum_i)$
  
​   可是我们发现,不同深度挑选的区间一旦有重复,就意味着有蚂蚁会同时刻到达终点,也就是早就会撞上,因此挑选的区间必须互相不重叠。
  
​   那么每个深度的区间开头位置往往取不到$i$,因为前面会顺延下来造成推移。
  
​   答案求的其实是最靠后的一个区间的结束位置,记每个深度为$i$的节点到达时间区间为$[start_i,start_i+sum_i)$,那么实际上$ans=max \{ start_i+sum_i-1\}$。
  
​   然而维护$start$太难了,下面证明实际上$ans=max\{i+a_i-1\},i>0$,其中$a_i$表示深度大于等于$i$的蚂蚁个数。
  
​   首先根节点的蚂蚁全部忽略掉不考虑。
  
​   设深度为$i$的蚂蚁们被分配到的区间是最后的。
  
  ​ 如果开头位置取到了$i$,那么$i+a_i-1$可以贡献到答案,并且发现其他情况时都不可能贡献到比这个还大的位置。
  
​   如果开头位置没有取到$i$,说明这个区间被顺延了,根据贪心策略,这个区间一定紧挨着上一个区间,以此类推形成一个连通块。设连通块最前端的区间是给$j$取的,而且此区间的开始位置一定是最好情况:刚好取到$j$,所以$j+a_j-1$可以贡献到$i$的结尾位置。
  
  ​ 所以用线段树维护一下$i+a_i-1$就好了。
    
```c++
#include <cstdio>
#include <set>
using namespace std;
const int N=100005,M=200005;
namespace IO{
const int SIZE=10000000;
char buffer[SIZE];
int pos;
void init(){fread(buffer,1,SIZE,stdin);}
char getch(){return buffer[pos++];}
int getInt(){
char c=getch(); int x=0,f=1;
while(c<'0'||c>'9'){if(c=='-')f=-f;c=getch();}
while('0'<=c&&c<='9'){x=x*10+c-'0';c=getch();}
return x*f;
}
}
int n,m,antcnt,a[N];
int h[N],tot,dep[N],maxdep;
struct Edge{int v,next;}e[M];
struct Data{
int x;
Data(){}
Data(int _x){x=_x;}
friend bool operator < (const Data &a,const Data &b){
if(dep[a.x]!=dep[b.x])
return dep[a.x]<dep[b.x];
return a.x<b.x;
}
};
set<Data> s;
inline int max(int x,int y){return x>y?x:y;}
inline void addEdge(int u,int v){e[++tot].v=v;e[tot].next=h[u];h[u]=tot;}
void dfs(int u,int fa){
maxdep=max(maxdep,dep[u]);
for(int i=h[u],v;i;i=e[i].next)
if((v=e[i].v)!=fa){
dep[v]=dep[u]+1;
dfs(v,u);
}
}
namespace SEG{/*{{{*/
int rt,sz,ch[N*2][2],maxs[N*2],tag[N*2];
void addtag(int u,int x){
maxs[u]+=x;
tag[u]+=x;
}
inline void pushup(int u){maxs[u]=max(maxs[ch[u][0]],maxs[ch[u][1]]);}
inline void pushdown(int u){
if(tag[u]){
addtag(ch[u][0],tag[u]);
addtag(ch[u][1],tag[u]);
tag[u]=0;
}
}
void build(int &u,int l,int r){
u=++sz;
maxs[u]=r;
if(l==r) return;
int mid=(l+r)>>1;
build(ch[u][0],l,mid);
build(ch[u][1],mid+1,r);
}
void modify(int u,int l,int r,int L,int R,int x){
if(L<=l&&r<=R){
addtag(u,x);
return;
}
pushdown(u);
int mid=(l+r)>>1;
if(L<=mid) modify(ch[u][0],l,mid,L,R,x);
if(mid<R) modify(ch[u][1],mid+1,r,L,R,x);
pushup(u);
}
int query(int u,int l,int r,int L,int R){
if(L<=l&&r<=R) return maxs[u];
pushdown(u);
int mid=(l+r)>>1;
if(R<=mid) return query(ch[u][0],l,mid,L,R);
else if(mid<L) return query(ch[u][1],mid+1,r,L,R);
else return max(query(ch[u][0],l,mid,L,mid),query(ch[u][1],mid+1,r,mid+1,R));
}
}/*}}}*/
int main(){
freopen("input.in","r",stdin);
IO::init();
n=IO::getInt();
m=IO::getInt();
for(int i=2;i<=n;i++){
int x=IO::getInt();
addEdge(x,i);
}
dfs(1,0);
int opt,x;
SEG::build(SEG::rt,0,maxdep);
for(int i=1;i<=m;i++){
opt=IO::getInt();
x=IO::getInt();
if(opt==1){
a[x]++; antcnt++;
if(a[x]==1) s.insert(Data(x));
}
else{
a[x]--; antcnt--;
if(!a[x]) s.erase(s.find(Data(x)));
}
if(x!=1)
SEG::modify(SEG::rt,0,maxdep,0,dep[x],opt==1?1:-1);
if(!antcnt||a[1]==antcnt)
printf("0\n");
else{
set<Data>::iterator pos=s.end();
pos--;
printf("%d\n",SEG::query(SEG::rt,0,maxdep,0,dep[(*pos).x]));
}
}
return 0;
}

Ants on tree的更多相关文章

  1. Educational Codeforces Round 7 E&period; Ants in Leaves 贪心

    E. Ants in Leaves 题目连接: http://www.codeforces.com/contest/622/problem/E Description Tree is a connec ...

  2. 【UVA 1411】 Ants (KM)

    Young naturalist Bill studies ants in school. His ants feed onplant-louses that live on apple trees. ...

  3. UVALive 4043 Ants

    KM   构图求最小权值匹配 保证最小的权值,所连的边一定是能够不相交的. Ants Time Limit: 3000MS   Memory Limit: Unknown   64bit IO For ...

  4. poj 2565 Ants &lpar;KM&plus;思维&rpar;

    Ants Time Limit: 5000MS   Memory Limit: 65536K Total Submissions: 4125   Accepted: 1258   Special Ju ...

  5. &lbrack;置顶&rsqb; Ants&lpar;Northeastern Europe 2007&rpar;

                                                                                      Ants Time Limit: 5 ...

  6. &lbrack;poj3565&rsqb;Ants

    [poj3565]Ants 标签(空格分隔):二分图 描述 Young naturalist Bill studies ants in school. His ants feed on plant-l ...

  7. POJ 3565 Ants 【最小权值匹配应用】

    传送门:http://poj.org/problem?id=3565 Ants Time Limit: 5000MS   Memory Limit: 65536K Total Submissions: ...

  8. POJ 3565 Ants(最佳完美匹配)

    Description Young naturalist Bill studies ants in school. His ants feed on plant-louses that live on ...

  9. HDU 4776 Ants(Trie&plus;优先队列)

    Ants Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 327680/327680 K (Java/Others) Total S ...

随机推荐

  1. servlet 生命周期

    Ò编写一个HelloWordServlet类

  2. Swift实战-豆瓣电台(五)播放音乐

    观看地址 http://v.youku.com/v_show/id_XNzMwODM0MzI0.html 在这节里面,我们简单学习了一下MediaPlayer的使用 引入媒体框架 import Med ...

  3. Sublime Text Snippets(代码片段)功能

    原文链接:http://www.bluesdream.com/blog/sublime-text-snippets-function.html 我们在编写代码的时候,总会遇到一些需要反复使用的代码片段 ...

  4. gradle的安装配置成功标志

    gradle主要位于AndroidStudio中 看我的目录 在环境变量里添加用户变量 GRADLE_HOME 然后在环境变量 path 中增加 %GRADLE_HOME%\bin;,如图所示 测试配 ...

  5. Java 存储时间戳的几种方式

    有时需要记录一下数据生成时间的时间戳,精确到秒,这里记录一下java存储时间戳字符串的几种方式 1.DateFormat private static final SimpleDateFormat s ...

  6. 【Java POI】1、Java POI的使用

    很多时候,一个软件应用程序需要生成Microsoft Excel文件格式的报告.有时,一个应用程序甚至希望将Excel文件作为输入数据.例如,一个公司开发的应用程序将财务部门需要所有输出生成自己的Ex ...

  7. 大数据技术之&lowbar;16&lowbar;Scala学习&lowbar;01&lowbar;Scala 语言概述

    第一章 Scala 语言概述1.1 why is Scala 语言?1.2 Scala 语言诞生小故事1.3 Scala 和 Java 以及 jvm 的关系分析图1.4 Scala 语言的特点1.5 ...

  8. Linux free命令使用及解析

    1. 命令格式 free [参数] 2. 命令功能 free 命令显示系统使用和空闲的内存情况,包括物理内存.交互区内存(swap)和内核缓冲区内存.共享内存将被忽略 3. 命令参数 -b 以Byte ...

  9. Django实战(17):ajax &excl;

    现在让我们来通过ajax请求后台服务.当然首选要实现后台服务.关于“加入购物车”,我们需要的服务是这样定义的: url:    http://localhost:8000/depotapp/API/c ...

  10. spring mvc 图片上传&comma;图片压缩、跨域解决、 按天生成文件夹 &comma;删除,限制为图片代码等相关配置

    spring mvc 图片上传,跨域解决 按天生成文件夹 ,删除,限制为图片代码,等相关配置 fs.root=data/ #fs.root=/home/dev/fs/ #fs.root=D:/fs/ ...