题面

解析

事实上,这应该是道树剖裸题了,

将已安装表示为\(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】软件包管理器的更多相关文章

  1. 【题解】NOI2015软件包管理器

    [题解][P2146 NOI2015]软件包管理器 实际上就是树链剖分板子题. 对于\(install\)操作,直接查询它到\(0\)节点有多少已经安装了的,再用总数减去它. 对于\(uninstal ...

  2. 题解 P2146 [NOI2015]软件包管理器

    P2146 [NOI2015]软件包管理器 感觉代码比其他题解更简洁qwq 树链剖分模板题 install x:将1~x的路径上的节点全部变成1(安装x需要先安装1~x) uninstall x:将x ...

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

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

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

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

  5. [UOJ#128][BZOJ4196][Noi2015]软件包管理器

    [UOJ#128][BZOJ4196][Noi2015]软件包管理器 试题描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管 ...

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

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

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

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

  8. [NOI2015]软件包管理器

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

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

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

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

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

随机推荐

  1. 查找担保圈-step7-提取未被包含过组的成员,得出结论

    USE [test] GO /****** Object: StoredProcedure [dbo].[p05_get_group_member_cleared] Script Date: 2019 ...

  2. django ajax MTV与MVC 多对多创建方式

    MTV与MVC MTV模型(django): M:模型层(models.py) T:templates V:views MVC模型: M:模型层(models.py) V:视图层(views.py) ...

  3. zabbix-自定义告警(二)

    实现自定义监控 文章引用:https://www.cnblogs.com/clsn/p/7885990.html#auto_id_28 一.实现自定义监控 说明zabbix自带模板Template O ...

  4. 经典算法,yuv与rgb互转,查表法,让你的软件飞起来

    代码的运算速度取决于以下几个方面 1. 算法本身的复杂度,比如MPEG比JPEG复杂,JPEG比BMP图片的编码复杂. 2. CPU自身的速度和设计架构 3. CPU的总线带宽 4. 您自己代码的写法 ...

  5. 第二十六篇 jQuery 学习8 遍历-父亲兄弟子孙元素

    jQuery 学习8 遍历-父亲兄弟子孙元素   jQuery遍历,可以理解为“移动”,使用“移动”还获取其他的元素.   什么意思呢?老师举一个例子: 班上30位同学,我是新来负责教这个班学生的老师 ...

  6. TensorFlow中CNN的两种padding方式“SAME”和“VALID”

    来源 dilation_rate为一个可选的参数,默认为1,这里我们可以先不管它. 整理一下,对于"VALID",输出的形状计算如下: new_height=new_width=⌈ ...

  7. 复制SD启动卡 生成新启动卡

    在已经有1张SD卡启动卡的情况下,如何复制出一张新卡: 1. 使用软件DiskGenius4.8.0->硬盘->备份分区表 备份 源sd卡分区信息 2. 使用软件DiskGenius4.8 ...

  8. vim 绑定到 source insight 快捷键

    1. optioons -> custom commands 2. 选择然后写入run命令: "D:\Program Files (x86)\Vim\vim74\gvim.exe&qu ...

  9. Java SE 核心 I

    Java SE 核心 I 1.Object类 在 Java 继承体系中,java.lang.Object 类位于顶端(是所有对象的直接或间接父类).如果一个类没有写 extends 关键字声明其父类, ...

  10. (十一)设置关闭多核cpu的核

    echo 0 > /sys/devices/system/cpu/cpu3/online 查看当前有哪些核心 cat  /sys/devices/system/cpu/online