题解 【NOI2015】软件包管理器
题面
解析
事实上,这应该是道树剖裸题了,
将已安装表示为\(1\),
那么只需要在线段树中记录一下区间中\(1\)的个数就行了.
在询问的时候,
如果是安装,就查询\(x\)到根节点,
卸载的话,就查询\(x\)的子树.
注意下细节就行了(最好整体位移一下).
感觉真的没什么讲的了qwq
上代码吧:
#include <iostream>
#include <cstdio>
#include <cstring>
#define ls(a) a<<1
#define rs(a) a<<1|1
using namespace std;
inline int read(){
int sum=0,f=1;char c=getchar();
while(c<'0'||c>'9'){if(c=='-') f=-1;c=getchar();}
while(c>='0'&&c<='9'){sum=sum*10+c-'0';c=getchar();}
return sum*f;
}
struct edge{int to,next;}e[100001];
struct node{int dep,top,fa,son,size,id;}a[100001];
struct tree{int l,r,val,tag;}t[400001];
int n,m;
int head[100001],cnt=0;
int tot=0;
inline void add(int x,int y){
e[++cnt]=(edge){head[x],y};head[x]=cnt;
}
void dfs1(int x,int fa){
a[x].dep=a[fa].dep+1;a[x].fa=fa;a[x].size=1;
for(int i=head[x];i;i=e[i].to){
int k=e[i].next;if(k==fa) continue;
dfs1(k,x);a[x].size+=a[k].size;
if(!a[x].son||a[a[x].son].size<a[k].size) a[x].son=k;
}
}
void dfs2(int x,int t){
a[x].id=++tot;a[x].top=t;
if(a[x].son) dfs2(a[x].son,t);
for(int i=head[x];i;i=e[i].to){
int k=e[i].next;if(k==a[x].fa||k==a[x].son) continue;
dfs2(k,k);
}
}
void pushup(int p){
t[p].val=t[ls(p)].val+t[rs(p)].val;
}
void pushdown(int p){
if(!t[p].tag) return ;
int ls=ls(p),rs=rs(p);
t[ls].tag=t[rs].tag=t[p].tag;
if(t[p].tag==2) t[p].tag=0;
t[ls].val=t[p].tag*(t[ls].r-t[ls].l+1);
t[rs].val=t[p].tag*(t[rs].r-t[rs].l+1);
t[p].tag=0;
}
void build(int p,int l,int r){
t[p].l=l;t[p].r=r;
if(l==r) return ;
int mid=(l+r)>>1;
build(ls(p),l,mid);build(rs(p),mid+1,r);
}
void change(int p,int l,int r,int opt){
if(t[p].l>=l&&t[p].r<=r){
t[p].val=opt*(t[p].r-t[p].l+1);t[p].tag=opt;
if(!opt) t[p].tag=2;
return ;
}
pushdown(p);
int mid=(t[p].l+t[p].r)>>1;
if(l<=mid) change(ls(p),l,r,opt);
if(r>mid) change(rs(p),l,r,opt);
pushup(p);
}
int ask1(int p,int l,int r){
if(t[p].l>=l&&t[p].r<=r) return t[p].val;
pushdown(p);
int mid=(t[p].l+t[p].r)>>1,ans=0;
if(l<=mid) ans+=ask1(ls(p),l,r);
if(r>mid) ans+=ask1(rs(p),l,r);
pushup(p);
return ans;
}
int ask0(int p,int l,int r){
if(t[p].l>=l&&t[p].r<=r) return t[p].r-t[p].l+1-t[p].val;
pushdown(p);
int mid=(t[p].l+t[p].r)>>1,ans=0;
if(l<=mid) ans+=ask0(ls(p),l,r);
if(r>mid) ans+=ask0(rs(p),l,r);
pushup(p);
return ans;
}
inline void work1(int x,int y,int opt){
int ans=ask1(1,a[x].id,a[x].id+a[x].size-1);
change(1,a[x].id,a[x].id+a[x].size-1,opt);
printf("%d\n",ans);
}
inline void work0(int x,int y,int opt){
int ans=0;
while(a[x].top!=a[y].top){
if(a[a[x].top].dep<a[a[y].top].dep) swap(x,y);
ans+=ask0(1,a[a[x].top].id,a[x].id);
change(1,a[a[x].top].id,a[x].id,opt);x=a[a[x].top].fa;
}
if(a[x].dep>a[y].dep) swap(x,y);
ans+=ask0(1,a[x].id,a[y].id);change(1,a[x].id,a[y].id,opt);
printf("%d\n",ans);
}
int main(){
n=read();
for(int i=2;i<=n;i++){int x=read()+1;add(x,i);}
m=read();
dfs1(1,0);dfs2(1,1);build(1,1,n);
for(int i=1;i<=m;i++){
string s;cin>>s;int x=read()+1;
if(s[0]=='u') work1(x,1,0);
else if(s[0]=='i') work0(x,1,1);
}
return 0;
}
题解 【NOI2015】软件包管理器的更多相关文章
- 【题解】NOI2015软件包管理器
[题解][P2146 NOI2015]软件包管理器 实际上就是树链剖分板子题. 对于\(install\)操作,直接查询它到\(0\)节点有多少已经安装了的,再用总数减去它. 对于\(uninstal ...
- 题解 P2146 [NOI2015]软件包管理器
P2146 [NOI2015]软件包管理器 感觉代码比其他题解更简洁qwq 树链剖分模板题 install x:将1~x的路径上的节点全部变成1(安装x需要先安装1~x) uninstall x:将x ...
- Bzoj 4196: [Noi2015]软件包管理器 树链剖分
4196: [Noi2015]软件包管理器 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 721 Solved: 419[Submit][Statu ...
- 【BZOJ4196】[Noi2015]软件包管理器 树链剖分
[Noi2015]软件包管理器 树链剖分 Description Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从 ...
- [UOJ#128][BZOJ4196][Noi2015]软件包管理器
[UOJ#128][BZOJ4196][Noi2015]软件包管理器 试题描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管 ...
- BZOJ 4196: [Noi2015]软件包管理器 [树链剖分 DFS序]
4196: [Noi2015]软件包管理器 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 1352 Solved: 780[Submit][Stat ...
- [BZOJ4196][NOI2015]软件包管理器
4196: [Noi2015]软件包管理器 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 1040 Solved: 603[Submit][Stat ...
- [NOI2015]软件包管理器
4621 [NOI2015]软件包管理器 题目等级 : 钻石 Diamond 题目描述 Description Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过 ...
- BZOJ_4196_[Noi2015]软件包管理器_树链剖分
BZOJ_4196_[Noi2015]软件包管理器_树链剖分 题意: Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助 ...
- bzoj 4196 [Noi2015]软件包管理器 (树链剖分+线段树)
4196: [Noi2015]软件包管理器 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 2852 Solved: 1668[Submit][Sta ...
随机推荐
- [转帖]InfluxDB 1.2.0安装及新旧版本的注意事项
InfluxDB 1.2.0安装及新旧版本的注意事项 http://haibing.org/245?zwlqby=npztq3 挺好的文章 很好的解决了 上一个文档里面 关于 web admin 的问 ...
- quartus ii 粗略使用教程
重复刚刚做过的下载程序,不选sof文件,选择jic文件 选择program config然后点击start,观察开发板,断电在开启后仍然有效果,想要擦除开发板flash文件,可以点击取消program ...
- Linux就该这么学——新手必须掌握的命令之文件目录管理命令组
touch命令 用途 : 用于创建空白文件或设置文件的时间 格式 : touch [选项] [文件] 参数 作用 -a 仅修改”读取时间”(atime) -m 仅修改”修改时间”(mtime) -d ...
- sqlalchemy定义mysql时间戳字段
update_time = Column(TIMESTAMP, nullable=False, comment='更新时间戳', server_default=text('CURRENT_TIMEST ...
- JSONObject 的使用
1. 导入依赖 这里以 20180813 的 json 版本为例 <dependency> <groupId>org.json</groupId> <arti ...
- maven工程下整合spring+mybatis报Mapped Statements collection does not contain value for spring-mybatis-user-get错误
在整合spring+mybatis报了下面的错误: Mapped Statements collection does not contain value for spring-mybatis-use ...
- element-ui table float类型数据排序失败
背景:对于16.88这样的数据,点击表头排序无效,仍然是乱序 解决办法:自定义排序方法,:sortable="true" :sort-mothod="xxxx" ...
- 【原创】大叔经验分享(62)kudu副本数量
kudu的副本数量是在表上设置,可以通过命令查看 # sudo -u kudu kudu cluster ksck $master ... Summary by table Name | RF | S ...
- JavaScript函数式编程——柯里化
柯里化原理 如何实现柯里化 柯里化的应用 一.柯里化原理 柯里化:在数学和计算机科学中,柯里化是一种使用多个参数的一个函数转换成一系列使用一个参数的函数的技术. 前端使用柯里化的用途主要就应该是简化代 ...
- Javascript 中apply call bind
在 javascript 中,call 和 apply 都是为了改变某个函数运行时的上下文(context)而存在的,换句话说,就是为了改变函数体内部 this 的指向. 先来一个例子: functi ...