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

题目描述
Linux用户和OSX用户一定对软件包管理器不会陌生。通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件源下载软件包,同时自动解决所有的依赖(即下载安装这个软件包的安装所依赖的其它软件包),完成所有的配置。Debian/Ubuntu使用的apt-get,Fedora/CentOS使用的yum,以及OSX下可用的homebrew都是优秀的软件包管理器。
你决定设计你自己的软件包管理器。不可避免地,你要解决软件包之间的依赖问题。如果软件包A依赖软件包B,那么安装软件包A以前,必须先安装软件包B。同时,如果想要卸载软件包B,则必须卸载软件包A。现在你已经获得了所有的软件包之间的依赖关系。而且,由于你之前的工作,除0号软件包以外,在你的管理器当中的软件包都会依赖一个且仅一个软件包,而0号软件包不依赖任何一个软件包。依赖关系不存在环(若有m(m≥2)个软件包A1,A2,A3,⋯,Am,其中A1依赖A2,A2依赖A3,A3依赖A4,……,A[m-1]依赖Am,而Am依赖A1,则称这m个软件包的依赖关系构成环),当然也不会有一个软件包依赖自己。
现在你要为你的软件包管理器写一个依赖解决程序。根据反馈,用户希望在安装和卸载某个软件包时,快速地知道这个操作实际上会改变多少个软件包的安装状态(即安装操作会安装多少个未安装的软件包,或卸载操作会卸载多少个已安装的软件包),你的任务就是实现这个部分。注意,安装一个已安装的软件包,或卸载一个未安装的软件包,都不会改变任何软件包的安装状态,即在此情况下,改变安装状态的软件包数为0。
输入格式
从文件manager.in中读入数据。
输入文件的第1行包含1个整数n,表示软件包的总数。软件包从0开始编号。
随后一行包含n−1个整数,相邻整数之间用单个空格隔开,分别表示1,2,3,⋯,n−2,n−1号软件包依赖的软件包的编号。
接下来一行包含1个整数q,表示询问的总数。之后q行,每行1个询问。询问分为两种:
install x:表示安装软件包x
uninstall x:表示卸载软件包x
你需要维护每个软件包的安装状态,一开始所有的软件包都处于未安装状态。
对于每个操作,你需要输出这步操作会改变多少个软件包的安装状态,随后应用这个操作(即改变你维护的安装状态)。
输出格式
输出到文件manager.out中。
输出文件包括q行。
输出文件的第i行输出1个整数,为第i步操作中改变安装状态的软件包数。

输入输出样例
输入 #1
7
0 0 0 1 1 5
5
install 5
install 6
uninstall 1
install 4
uninstall 0

输出 #1
3
1
3
2
3

输入 #2
10
0 1 2 1 3 0 0 3 2
10
install 0
install 3
uninstall 2
install 7
install 5
install 9
uninstall 9
install 4
install 1
install 9

输出 #2
1
3
2
1
3
1
1
1
0
1

说明/提示
在这里插入图片描述

解释:如果存在我们记为1,不存在我们记为0,这样我们就转行成树上求和问题了,直接上树链剖分模板

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define lson rt<<1
#define rson rt<<1|1
#define ll long long
const ll N=100003;
const ll inf=-100000009;
using namespace std;
ll n,idx,dfn[N],seq[N],fa[N],dep[N],top[N],sz[N],hvy[N];
ll dp[N]={0};
ll tree[8*N]={0},val[N]={0};
ll a[N]={0};
ll lazy[8*N]={0};
ll tot,lnk[N],ter[4*N],nxt[4*N];
void add(ll u,ll v){
    ter[++tot]=v,nxt[tot]=lnk[u],lnk[u]=tot;
}
void dfs1(ll u,ll f){
    fa[u]=f,dep[u]=dep[f]+1,sz[u]=1,hvy[u]=top[u]=0;
    for(ll i=lnk[u];i;i=nxt[i]){
        ll v=ter[i];
        if(v==f) continue;
        dp[v]=dp[u]+1;
        dfs1(v,u);
        sz[u]+=sz[v];
        if(sz[hvy[u]]<sz[v]) hvy[u]=v;
    }
}
void dfs2(ll u,ll tp) {
    dfn[u]=++idx,seq[idx]=u,top[u]=tp;
    if(!hvy[u]) return;
    dfs2(hvy[u],tp);
    for(ll i=lnk[u];i;i=nxt[i]) {
        ll v=ter[i];
        if(v==fa[u]||v==hvy[u]) continue;
        dfs2(v,v);
    }
}
void pushup(ll rt) {
    tree[rt]=tree[lson]+tree[rson];
}
void pushdown(ll rt,ll l,ll r){
    ll mid=(l+r)/2;
    if(lazy[rt]==-1) return;
    tree[lson]=lazy[rt]*(mid-l+1);
    tree[rson]=lazy[rt]*(r-mid);
    lazy[lson]=lazy[rt];
    lazy[rson]=lazy[rt];
    lazy[rt]=-1;
    return;
}
void build(ll rt,ll l,ll r) {
    if(l==r) {
        tree[rt]=val[seq[l]];
        lazy[rt]=-1;
        return;
    }
    ll mid=(l+r)>>1;
    build(lson,l,mid);
    build(rson,mid+1,r);
    pushup(rt);
}
void modify(ll L,ll R,ll rt,ll l,ll r,ll val) {
    pushdown(rt,l,r);
    if(L<=l&&r<=R){
        tree[rt]=val*(r-l+1);
        lazy[rt]=val;
        return;
    }
    ll mid=(l+r)>>1;
    if(L<=mid) modify(L,R,lson,l,mid,val);
    if(R>mid) modify(L,R,rson,mid+1,r,val);
    pushup(rt);
}
void chainmodify(ll u,ll v,ll val) {
    for(ll fu=top[u],fv=top[v];fu^fv;u=fa[fu],fu=top[u]){
        if(dep[fu]<dep[fv]) swap(u,v),swap(fu,fv);
        modify(dfn[fu],dfn[u],1,1,n,val);
    }
    if(dep[u]>dep[v]) swap(u,v);
    modify(dfn[u],dfn[v],1,1,n,val);
    return ;
}
ll querySUM(ll x,ll y,ll rt,ll l,ll r){
    pushdown(rt,l,r);
    if(x>y) return 0;
    if(x<=l&&r<=y) return tree[rt];
    ll mid=(l+r)>>1;
    ll res=0;
    if(x<=mid) res+=querySUM(x,y,lson,l,mid);
    if(mid<y) res+=querySUM(x,y,rson,mid+1,r);
    return res;
}
ll chainQuerySUM(ll u,ll v){
    ll res=0;
    for(ll fu=top[u],fv=top[v];fu^fv;u=fa[fu],fu=top[u]){
        if(dep[fu]<dep[fv]) swap(u,v),swap(fu,fv);
        res+=querySUM(dfn[fu],dfn[u],1,1,n);
    }
    if(dep[u]>dep[v]) swap(u,v);
    res+=querySUM(dfn[u],dfn[v],1,1,n);
    return res;
}
int main(){
    scanf("%lld",&n);
    for(int i=2;i<=n;i++) scanf("%lld",&a[i]);
    for(int i=2;i<=n;++i){
        add(i,a[i]+1),add(a[i]+1,i);
    }
    dp[1]=1;
    dfs1(1,0),dfs2(1,1),build(1,1,n);
    int q=0;scanf("%d",&q);
    while(q--){
        char cmd[123];int x;scanf("%s%d",cmd,&x);x++;
        ll ret=0;
        if(cmd[0]=='i'){
            ret=dp[x]-chainQuerySUM(1,x);
            chainmodify(1,x,1);
        }else{
            ret=querySUM(dfn[x],dfn[x]+sz[x]-1,1,1,n);
            modify(dfn[x],dfn[x]+sz[x]-1,1,1,n,0);
        }
        cout<<ret<<endl;
    }
    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值