显然链剖 然而只询问到根的信息,不用管lca,要好些很多(虽然我没那么写)

对于安装 查询和维护到根路径

对于卸载 查询和维护子树信息

因为链剖本身是用dfs序建的线段树,所以使得查询和修改子树非常方便。

 #include<iostream>
#include<cstring>
#include<cstdlib>
#include<cstdio>
#include<algorithm> using namespace std; const int Maxn=;
int n;
struct Data{
int num[];
Data(int a=,int b=) {num[]=a,num[]=b;}
Data operator + (const Data&rhs) const {
return Data(num[]+rhs.num[],num[]+rhs.num[]);
}
}; struct SegmentTree{
Data da[Maxn*];
int tag[Maxn*];
int lft,rgt,w; void add_tag(int s,int l,int r,int w) {
if(w==-) return;
da[s].num[w]=r-l+;
da[s].num[w^]=;
tag[s]=w;
} void down(int s,int l,int r) {
if(tag[s]==- || l==r) return;
int mid=(l+r)>>;
add_tag(s<<,l,mid,tag[s]);
add_tag(s<<|,mid+,r,tag[s]);
tag[s]=-;
} void change(int s,int l,int r) {
if(lft<=l && r<=rgt) {
add_tag(s,l,r,w);
return;
}
down(s,l,r);
int mid=(l+r)>>;
if(lft<=mid) change(s<<,l,mid);
if(mid<rgt) change(s<<|,mid+,r);
da[s] = da[s<<] + da[s<<|];
}
Data query(int s,int l,int r) {
if(lft<=l&&r<=rgt) return da[s];
down(s,l,r);
int mid=(l+r)>>;
if(rgt<=mid) return query(s<<,l,mid);
if(mid<lft) return query(s<<|,mid+,r);
return query(s<<,l,mid) + query(s<<|,mid+,r);
}
void build(int s,int l,int r) {
if(l==r) {
da[s]=Data(,);
return;
}
int mid=(l+r)>>;
build(s<<,l,mid);
build(s<<|,mid+,r);
da[s]=da[s<<]+da[s<<|];
}
}seg; int sz[Maxn],fa[Maxn],son[Maxn],dep[Maxn]; struct Edge {
int to;
Edge*next;
Edge(int to=,Edge*next=):to(to),next(next) {}
}pool[Maxn*],*pis=pool,*fir[Maxn]; void AddEdge(int from,int to) {
*pis=Edge(to,fir[from]);fir[from]=pis++;
*pis=Edge(from,fir[to]);fir[to]=pis++;
} void dfs(int u) {
sz[u] = ;
son[u] = ;
for(Edge*p=fir[u];p;p=p->next) {
int v=p->to;
if(v==fa[u]) continue;
fa[v]=u;
dep[v]=dep[u]+;
dfs(v);
sz[u]+=sz[v];
if(sz[v]>sz[son[u]]) son[u]=v;
}
} int pos[Maxn],top[Maxn],tm,end[Maxn]; void build(int u,int pre) {
top[u] = pre;
pos[u] = ++tm;
if(son[u]) build(son[u],pre);
for(Edge*p=fir[u];p;p=p->next) {
int v=p->to;
if(v==fa[u] || v==son[u]) continue;
build(v,v);
}
end[u]=tm;
} Data query(int a,int b) {
int ta=top[a],tb=top[b];
Data ret;
for(;ta!=tb;a=fa[ta],ta=top[a]) {
if(dep[ta]<dep[tb]) swap(a,b) ,swap(ta,tb);
seg.lft=pos[ta],seg.rgt=pos[a];
ret=ret+seg.query(,,n);
}
if(dep[a]>dep[b]) swap(a,b);
seg.lft=pos[a],seg.rgt=pos[b];
Data tmp=seg.query(,,n);
return ret+tmp;
} void change(int a,int b,int c) {
int ta=top[a],tb=top[b];
Data ret;
for(;ta!=tb;a=fa[ta],ta=top[a]) {
if(dep[ta]<dep[tb]) swap(a,b) ,swap(ta,tb);
seg.lft=pos[ta],seg.rgt=pos[a];seg.w=c;
seg.change(,,n);
}
if(dep[a]>dep[b]) swap(a,b);
seg.lft=pos[a],seg.rgt=pos[b],seg.w=c;
return seg.change(,,n);
} void init() {
memset(seg.tag,-,sizeof seg.tag);
scanf("%d",&n);
for(int x,i=;i<n;i++) {
scanf("%d",&x);
AddEdge(i+,x+);
}
dfs();
build(,);
seg.build(,,n); // seg.lft=1,seg.rgt=n,seg.w=0;change(1,1,n);
} void work() {
int m,x;
char opt[];
for(scanf("%d",&m);m--;) {
scanf("%s%d",opt,&x);x++;
if(opt[]=='i') {
printf("%d\n",query(x,).num[]);
change(x,,);
}else {
seg.lft=pos[x],seg.rgt=end[x],seg.w=;
printf("%d\n",seg.query(,,n).num[]);
seg.change(,,n);
}
}
} int main() {
freopen("manager.in","r",stdin);
freopen("manager.out","w",stdout); init();
work(); return ;
}

NOI2015 软件包管理器 manager的更多相关文章

  1. 洛谷 P2146 [NOI2015]软件包管理器 树链剖分

    目录 题面 题目链接 题目描述 输入输出格式 输入格式: 输出格式: 输入输出样例 输入样例#1: 输出样例#1: 输入样例#2: 输出样例#2: 说明 说明 思路 AC代码 总结 题面 题目链接 P ...

  2. BZOJ 4196: [Noi2015]软件包管理器 [树链剖分 DFS序]

    4196: [Noi2015]软件包管理器 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 1352  Solved: 780[Submit][Stat ...

  3. [BZOJ4196][NOI2015]软件包管理器

    4196: [Noi2015]软件包管理器 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 1040  Solved: 603[Submit][Stat ...

  4. Bzoj 4196: [Noi2015]软件包管理器 树链剖分

    4196: [Noi2015]软件包管理器 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 721  Solved: 419[Submit][Statu ...

  5. [NOI2015]软件包管理器

    4621 [NOI2015]软件包管理器  题目等级 : 钻石 Diamond   题目描述 Description Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过 ...

  6. BZOJ_4196_[Noi2015]软件包管理器_树链剖分

    BZOJ_4196_[Noi2015]软件包管理器_树链剖分 题意: Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助 ...

  7. bzoj 4196 [Noi2015]软件包管理器 (树链剖分+线段树)

    4196: [Noi2015]软件包管理器 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 2852  Solved: 1668[Submit][Sta ...

  8. 洛谷 P2146 [NOI2015]软件包管理器 解题报告

    P2146 [NOI2015]软件包管理器 题目描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件源下载软 ...

  9. 【BZOJ4196】[Noi2015]软件包管理器 树链剖分

    [Noi2015]软件包管理器 树链剖分 Description Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从 ...

随机推荐

  1. iOS中常用的正则表达式

    iOS常用正则表达式 正则表达式用于字符串处理.表单验证等场合,实用高效.现将一些常用的表达式收集于此,以备不时之需. 匹配中文字符的正则表达式: [\u4e00-\u9fa5]评注:匹配中文还真是个 ...

  2. Chrome调试(转)

    原文地址:http://blog.csdn.net/chenmoquan/article/details/44943245#comments 觉得写的很适合web开发的新手 Chrome 的开发者工具 ...

  3. linux 监控系统缓存和cpu

    a=`free |head -n 2 |tail -n 1 |awk '{print $7}'`if [ $a -ge 900000 ];then     sync && echo 1 ...

  4. dedecms flag标签属性

    头条[h] flag='h' 推荐[c] flag='c' 幻灯[f] flag='f' 特荐[a] flag='a' 滚动[s] flag='s' 加粗[b] flag='b' 图片[p] flag ...

  5. php多条件组合查询

    1. 通过表单把查询条件提交到php文件中,在文件中以post的形式得到传送过来的条件. 2. 把传过来的查询条件赋给变量. 3. 判断如果查询条件非空,则拼接查询sql. 大体如下: 1. < ...

  6. Centos下安装配置LAMP(Linux+Apache+MySQL+PHP)

    Centos下安装配置LAMP(Linux+Apache+MySQL+PHP)   关于LAMP的各种知识,还请大家自行百度谷歌,在这里就不详细的介绍了,今天主要是介绍一下在Centos下安装,搭建一 ...

  7. Python自动化运维之23、Dom

    文档对象模型(Document Object Model,DOM)是一种用于HTML和XML文档的编程接口.它给文档提供了一种结构化的表示方法,可以改变文档的内容和呈现方式.最为关心的是,DOM把网页 ...

  8. Centos yum install

    http://wiki.centos.org/TipsAndTricks/BrokenVserver centos mirror:  http://mirror.centos.org/centos/6 ...

  9. UFLDL实验报告2:Sparse Autoencoder

    Sparse Autoencoder稀疏自编码器实验报告 1.Sparse Autoencoder稀疏自编码器实验描述 自编码神经网络是一种无监督学习算法,它使用了反向传播算法,并让目标值等于输入值, ...

  10. UIControl-IOS开发

    UIControl-IOS开发   UIKit提供了一组控件:UISwitch开关.UIButton按钮.UISegmentedControl分段控件.UISlider滑块.UITextField文本 ...