真没有想到,这竟然会是一道NOI的原题,听RQY说,这套题是北大出的,北大脑抽认为树剖很难。。。

只恨没有早学几年OI,只A这一道题也可以出去吹自己一A了NOI原题啊

好了,梦该醒了,我们来看题

以后放链接不放题面了,洛谷题面直接拷出来总是很迷

我是题面

读完题,我们会发现,这道题,好像是,树剖???裸题???

没错,就是裸题

题目要求两种操作,先说第一种

安装程序\(x\),并输出修改状态的程序数目。

就是从\(x\)一直查询+更新到根节点呗,节点数目减去已安装的数目就是答案,先查询,再更新,当然,操作和到一起也可以

第二种操作更无语,卸载程序\(x\)并输出修改状态的程序数目

说白了也就是看\(x\)的子树上安装了几个程序呗

那么线段树上,我们用1表示已安装,0表示未安装,一切就很显然了

上代码

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cctype>
#define ll long long
#define gc() getchar()
#define maxn 100005
using namespace std; inline ll read(){
ll a=0;int f=0;char p=gc();
while(!isdigit(p)){f|=p=='-';p=gc();}
while(isdigit(p)){a=(a<<3)+(a<<1)+(p^48);p=gc();}
return f?-a:a;
}
void write(ll a){
if(a>9)write(a/10);
putchar(a%10+'0');
}
int n,m; int tot,head[maxn];
struct ahaha1{
int to,next;
}e[maxn<<1];
inline void add(int u,int v){
e[tot].to=v;e[tot].next=head[u];head[u]=tot++;
} int f[maxn],sz[maxn],son[maxn],dep[maxn];
void dfs(int u,int fa){
int maxa=0;sz[u]=1;
for(int i=head[u];~i;i=e[i].next){
int v=e[i].to;if(v==fa)continue;
f[v]=u;dep[v]=dep[u]+1;
dfs(v,u);sz[u]+=sz[v];
if(sz[v]>maxa)maxa=sz[v],son[u]=v;
}
}
int top[maxn],in[maxn],b[maxn];
void dfs(int u,int fa,int topf){
top[u]=topf;in[u]=++tot;b[tot]=u;
if(!son[u])return;
dfs(son[u],u,topf);
for(int i=head[u];~i;i=e[i].next){
int v=e[i].to;if(v==fa||v==son[u])continue;
dfs(v,u,v);
}
} #define lc p<<1
#define rc p<<1|1
struct ahaha2{
int v,lz;
ahaha2(){
lz=-1;
}
}t[maxn<<2];
inline void pushup(int p){
t[p].v=t[lc].v+t[rc].v;
}
inline void pushdown(int p,int l,int r){
int m=l+r>>1;
t[lc].v=t[p].lz*(m-l+1);t[lc].lz=t[p].lz;
t[rc].v=t[p].lz*(r-m);t[rc].lz=t[p].lz;
t[p].lz=-1;
}
ll query1(int p,int l,int r,int L,int R){ //安装程序的操作
if(l>R||r<L)return 0;
if(L<=l&&r<=R){int v=t[p].v;t[p].v=r-l+1;t[p].lz=1;return v;} //通过先存储返回值然后进行修改来合并操作
int m=l+r>>1;if(t[p].lz!=-1)pushdown(p,l,r);
int l1=query1(lc,l,m,L,R),r1=query1(rc,m+1,r,L,R); //同上
pushup(p);
return l1+r1;
}
ll query2(int p,int l,int r,int L,int R){ //卸载程序的操作
if(l>R||r<L)return 0;
if(L<=l&&r<=R){int v=t[p].v;t[p].v=0;t[p].lz=0;return v;}
int m=l+r>>1;if(t[p].lz!=-1)pushdown(p,l,r);
int l1=query2(lc,l,m,L,R),r1=query2(rc,m+1,r,L,R);
pushup(p);
return l1+r1;
} inline void solve_1(){
int x=read(),ans1=0,ans2=0;
while(top[x]){
ans1+=in[x]-in[top[x]]+1; //别忘了+1
ans2+=query1(1,1,n,in[top[x]],in[x]);
x=f[top[x]];
}
ans1+=in[x]-in[0]+1;
ans2+=query1(1,1,n,in[0],in[x]);
write(ans1-ans2);putchar('\n');
}
inline void solve_2(){
int x=read();
write(query2(1,1,n,in[x],in[x]+sz[x]-1));putchar('\n'); //直接子树比安装简单多了
} int main(){memset(head,-1,sizeof head);
n=read();
for(int i=1;i<n;++i){
int a=read();
add(a,i);
}
tot=0;dfs(0,-1);dfs(0,-1,0);
m=read();
string z;
for(int i=1;i<=m;++i){
cin>>z;
switch(z[0]){
case 'i':solve_1();break;
case 'u':solve_2();break;
}
}
return 0;
}

洛谷 P2146 [NOI2015]软件包管理器的更多相关文章

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

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

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

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

  3. 洛谷 P2146 [NOI2015]软件包管理器 (树链剖分模板题)

    题目描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件源下载软件包,同时自动解决所有的依赖(即下载安装这个 ...

  4. 洛谷P2146 [NOI2015]软件包管理器

    https://www.luogu.org/problemnew/show/P2146 传送门 简单的树链剖分......维护下当前安装了多少个包......修改后查询下就行了......附上极其丑陋 ...

  5. 洛谷P2146 [NOI2015]软件包管理器 题解 树链剖分+线段树

    题目链接:https://www.luogu.org/problem/P2146 本题涉及算法: 树链剖分: 线段树(区间更新及求和,涉及懒惰标记) 然后对于每次 install x ,需要将 x 到 ...

  6. 洛谷 pP2146 [NOI2015]软件包管理器

    题目的传送门 题目描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件源下载软件包,同时自动解决所有的依赖( ...

  7. 洛谷 2146 [NOI2015]软件包管理器

    [题解] 每个软件只依赖另一个软件,且依赖关系不构成环,那么很容易想到这是树形结构. 我们用1表示以安装,用0表示未安装或已卸载:那么安装一个软件,就是把它到树根的路径上所有的点都改为1:卸载一个软件 ...

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

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

  9. P2146 [NOI2015]软件包管理器

    题目链接:https://www.luogu.org/problemnew/show/P2146 题目描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安 ...

随机推荐

  1. pager-taglib分页注意事项

    必须先导包,尤其是 jsp 这种工具类和标签库的

  2. Q&As:1.cocos2d-html5如何获得鼠标划过事件

    不喜欢按部就班学东西,感觉各种框架各种技术就应该是拿到手用的,这应该是导致我现在学了这么多却没一样精通的缘故吧. 发现自己喜欢在QQ群回答一些菜鸟的问题,就算自己不清楚也会乐意看代码帮助解决╮(╯_╰ ...

  3. 我们一起学习WCF 第二篇WCF承载多个接口

    前言:现在王大叔养了大批猪,赚了很多钱.但是最近发现养鸡也可以赚很多钱,他就像扩展业务开始养鸡.又过两年他发现市场对狗的需求量很大,他开始养狗.那么他改怎么做呢,不可能去修改猪住的地方把鸭子和狗放里面 ...

  4. Mac下重置MySQL密码

    第一步要先停止掉mysql服务: brew services stop mysql 第二步查看mysql安装路径: brew info mysql //我这里是brew管理的所以我用brew查看mys ...

  5. PHP性能优化 -理论篇

    什么情况下,遇到了PHP性能问题?    1 PHP语法使用的不恰当    2 使用PHP语言做不了它不擅长做的事    3 用php语言连接的服务不给力    4 PHP自身的短板    5 我也不 ...

  6. Rest-Assured 测试框架

    Rest-Assured 是一个测试 Restful Web Service 的 Java 类库,我们能够测试各种各样的请求组合,依次测试核心业务逻辑的不同组合. 它是通过发送特定的rest api, ...

  7. 亚马逊:PS4和Xbox One实在太火

    圣诞节刚刚结束,当实体零售商在抱怨坑爹的天气让自己节日生意变得冷清的同时,在线零售商们却依旧赚的盆满钵满. 亚马逊近日表示,今年节日期间的零售工作非常不错,新一代游戏机更是最大的亮点.据销售统计,在圣 ...

  8. hadoop的safemode 安全模式

    hadoop启动检查副本块数,就会进入safemode safemode的相关情况 虽然不能进行修改文件的操作,但是可以浏览目录结构.查看文件内容的. 在命令行下是可以控制安全模式的进入.退出和查看的 ...

  9. java-sun.misc.BASE64Decode AccessException

    在使用sun.misc中base64类时,eclipse可能会报找不到Access异常 只需要修改一下访问方式即可,如下: 右键项目->属性->Javabulid path->jre ...

  10. [Ubuntu] sogou中文输入法安装

    I install sogou 中文输入法 successfully, after following below steps: 1. install sogou pingyin by deb pac ...