洛谷 - P2146 - 软件包管理器 - 重链剖分
https://www.luogu.org/problem/P2146
继续重链剖分。
这里好像很好懂,每次安装软件就区间改值赋值整个路径是1,然后比较前后的sum值变化就可以了。事实上后一次的sum值必定是dep。卸载的话,依赖它的都没了,相当于清空整棵子树。
#include<bits/stdc++.h>
#define lc (o<<1)
#define rc (o<<1|1)
typedef long long ll;
using namespace std;
const int MAXN = 100000 + 5;
int dep[MAXN], siz[MAXN],  son[MAXN], fa[MAXN], top[MAXN], tid[MAXN], rnk[MAXN], cnt;
int n, m, r;
int head[MAXN], etop;
struct Edge {
    int v, next;
} e[MAXN * 2];
inline void init(int n) {
    etop = 0;
    memset(head, -1, sizeof(head[0]) * (n + 1));
}
inline void addedge(int u, int v) {
    e[++etop].v = v;
    e[etop].next = head[u];
    head[u] = etop;
    e[++etop].v = u;
    e[etop].next = head[v];
    head[v] = etop;
}
struct SegmentTree {
    int sum[MAXN * 4];
    int lz[MAXN * 4];//lz=1,set to 1;lz=-1,set to 0;
    void pushup(int o) {
        sum[o] = sum[lc] + sum[rc];
    }
    void pushdown(int o, int l, int r) {
        if(lz[o]) {
            lz[lc] = lz[o];
            lz[rc] = lz[o];
            int m = l + r >> 1;
            if(lz[o] == 1) {
                sum[lc] = (m - l + 1);
                sum[rc] = (r - m) ;
            } else {
                sum[lc] = 0;
                sum[rc] = 0;
            }
            lz[o] = 0;
        }
    }
    void update(int o, int l, int r, int ql, int qr, int v) {
        if(ql <= l && r <= qr) {
            lz[o] = v;
            if(v == 1)
                sum[o] = r - l + 1;
            else
                sum[o] = 0;
        } else {
            pushdown(o, l, r);
            int m = (l + r) >> 1;
            if(ql <= m)
                update(lc, l, m, ql, qr, v);
            if(qr >= m + 1)
                update(rc, m + 1, r, ql, qr, v);
            pushup(o);
        }
    }
    int query(int o, int l, int r, int ql, int qr) {
        if(ql <= l && r <= qr) {
            return sum[o];
        } else {
            pushdown(o, l, r);
            int m = (l + r) >> 1;
            int res = 0;
            if(ql <= m)
                res += query(lc, l, m, ql, qr);
            if(qr >= m + 1)
                res += query(rc, m + 1, r, ql, qr);
            return res;
        }
    }
} st;
void init1() {
    dep[r] = 1;
}
void dfs1(int u, int t) {
    siz[u] = 1, son[u] = -1, fa[u] = t;
    for(int i = head[u]; i != -1; i = e[i].next) {
        int v = e[i].v;
        if(v == t)
            continue;
        dep[v] = dep[u] + 1;
        dfs1(v, u);
        siz[u] += siz[v];
        if(son[u] == -1 || siz[v] > siz[son[u]])
            son[u] = v;
    }
}
void init2() {
    cnt = 0;
}
void dfs2(int u, int t) {
    top[u] = t;
    tid[u] = ++cnt;
    rnk[cnt] = u;
    if(son[u] == -1)
        return;
    dfs2(son[u], t);
    for(int i = head[u]; i != -1; i = e[i].next) {
        int v = e[i].v;
        if(v == fa[u] || v == son[u])
            continue;
        dfs2(v, v);
    }
}
int query1(int u, int v) {
    int ret = 0;
    for(int tu = top[u], tv = top[v]; tu != tv; u = fa[tu], tu = top[u]) {
        if(dep[tu] < dep[tv])
            swap(u, v), swap(tu, tv);
        ret += st.query(1, 1, n, tid[tu], tid[u]);
    }
    if(dep[u] > dep[v])
        swap(u, v);
    ret += st.query(1, 1, n, tid[u], tid[v]);
    return ret;
}
int query2(int u) {
    return st.query(1, 1, n, tid[u], tid[u]+siz[u]-1);
}
void update1(int u, int v, int val) {
    for(int tu = top[u], tv = top[v]; tu != tv; u = fa[tu], tu = top[u]) {
        if(dep[tu] < dep[tv])
            swap(u, v), swap(tu, tv);
        st.update(1, 1, n, tid[tu], tid[u], val);
    }
    if(dep[u] > dep[v])
        swap(u, v);
    st.update(1, 1, n, tid[u], tid[v], val);
    return;
}
void update2(int u, int val) {
    return st.update(1, 1, n, tid[u], tid[u]+siz[u]-1,val);
}
void install() {
    int x;
    scanf("%d", &x);
    int sum1 = query1(0, x);
    update1(0, x, 1);
    int sum2 = query1(0,x);
    printf("%d\n", sum2 - sum1);
    return ;
}
void uninstall() {
    int x;
    scanf("%d", &x);
    int sum1 = query2(x);
    update2(x, -1);
    printf("%d\n", sum1);
    return;
}
int main() {
#ifdef Yinku
    freopen("Yinku.in", "r", stdin);
#endif // Yinku
    scanf("%d", &n);
    init(n);
    for(int i = 1, u; i <= n - 1; ++i) {
        scanf("%d", &u);
        addedge(i, u);
    }
    r = 0;
    init1();
    dfs1(r, -1);
    init2();
    dfs2(r, r);
    char s[50];
    scanf("%d", &m);
    for(int i = 1; i <= m; ++i) {
        scanf("%s", s);
        switch(s[0]) {
            case 'i':
                install();
                break;
            case 'u':
                uninstall();
                break;
        }
    }
    return 0;
}
												
											洛谷 - P2146 - 软件包管理器 - 重链剖分的更多相关文章
- 洛谷 [P2146] 软件包管理器
		
树剖 将一个软件是否安装,看作是sum数组的0或1,对于每个操作前后sum[1]的变化,就是所求 #include <iostream> #include <cstdio> # ...
 - 【算法学习】【洛谷】树链剖分 & P3384 【模板】树链剖分 P2146 软件包管理器
		
刚学的好玩算法,AC2题,非常开心. 其实很早就有教过,以前以为很难就没有学,现在发现其实很简单也很有用. 更重要的是我很好调试,两题都是几乎一遍过的. 介绍树链剖分前,先确保已经学会以下基本技巧: ...
 - Luogu P2146 软件包管理器(树链剖分+线段树)
		
题意 给定\(n\)个软件包,每个软件包都有一个依赖软件包,安装一个软件包必须安装他的依赖软件包,卸载一个软件包必须先卸载所有依赖于它的软件包.给定\(m\)此操作,每次一个操作\(install/u ...
 - 【Luogu】P2146软件包管理器(树链剖分)
		
题目链接 上午跟rqy学了一道超难的概率题,准备颓一会,于是水了这么一道水题. 话说这题真的是模板啊.数据范围正好,描述特别贴近(都不给你绕弯子的),连图都给你画出来,就差题目描述加一句“树链剖分模板 ...
 - NOI2015 软件包管理器(树链剖分+线段树)
		
P2146 软件包管理器 题目描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件源下载软件包,同时自动解决 ...
 - 洛谷 P2146 [NOI2015]软件包管理器 (树链剖分模板题)
		
题目描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件源下载软件包,同时自动解决所有的依赖(即下载安装这个 ...
 - 洛谷 P2146 [NOI2015]软件包管理器 树链剖分
		
目录 题面 题目链接 题目描述 输入输出格式 输入格式: 输出格式: 输入输出样例 输入样例#1: 输出样例#1: 输入样例#2: 输出样例#2: 说明 说明 思路 AC代码 总结 题面 题目链接 P ...
 - 洛谷 P2146 [NOI2015]软件包管理器 解题报告
		
P2146 [NOI2015]软件包管理器 题目描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件源下载软 ...
 - AC日记——软件包管理器 洛谷 P2416
		
题目描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件源下载软件包,同时自动解决所有的依赖(即下载安装这个 ...
 
随机推荐
- Python操作 Excel表格
			
python 读写 excel 有好多选择,但是,方便操作的库不多,在我尝试了几个库之后,我觉得两个比较方便的库分别是 xlrd/xlwt.openpyxl. 我使用openpyxl 安装: pip ...
 - Java面试之基础篇(2)
			
11.是否可以从一个static方法内部发出对非static方法的调用? 不可以.因为非static方法是要与对象关联在一起的,必须创建一个对象后,才可以在该对象上进行方法调用,而static方法调用 ...
 - 【JavaMail】JavaMail整合RabbitMq发送邮件案例
			
前言 Linux安装RabbitMQ:https://www.cnblogs.com/jxd283465/p/11975094.html SpringBoot整合RabbitMQ:https://ww ...
 - 一、Nginx常见问题
			
1.相同server_name多个虚拟主机优先级访问 最先读取哪个配置文件,就访问那个的网页 2.location匹配优先级 相同location,会被后面的覆盖 匹配优先级更高的,找后面的 = ...
 - (23)C++/Python项目练习一
			
逆转字符串——输入一个字符串,将其逆转并输出. Python: def rev(s): return (s[::-1]) s =input("请输入一个字符串:") a = rev ...
 - 把Vim改装成一个IDE编程环境
			
一:安装中文帮助手册 1. 打开一个终端 2.下载vimcdoc-1.5.0.tar.gz 下载地址: http://nchc.dl.sourceforge.net/source ...
 - cdn for js library
			
https://cdnjs.com/libraries/jquery https://cdnjs.cloudflare.com/ajax/libs/jquery/3.4.1/jquery.js htt ...
 - MySQL闪回工具之myflash 和 binlog2sql
			
MySQL闪回工具之:binlog2sql https://github.com/danfengcao/binlog2sql MYSQL Binglog分析利器:binlog2sql使用详解 :h ...
 - 四、robotframework生成几种随机数
			
1.random()生成0<=n<1之间的随机实数--它会生成一个随机的浮点数,范围是在0.0~1.0之间.: ${num} evaluate random.random() ra ...
 - ngTemplateOutlet递归的问题
			
今天尝试通过 ng-template 加 ngTemplateOutlet实现一个递归的菜单.但是遇到一个问题:NullInjectorError: No provider for NzMenuDir ...