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 - 软件包管理器 - 重链剖分的更多相关文章

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

    树剖 将一个软件是否安装,看作是sum数组的0或1,对于每个操作前后sum[1]的变化,就是所求 #include <iostream> #include <cstdio> # ...

  2. 【算法学习】【洛谷】树链剖分 & P3384 【模板】树链剖分 P2146 软件包管理器

    刚学的好玩算法,AC2题,非常开心. 其实很早就有教过,以前以为很难就没有学,现在发现其实很简单也很有用. 更重要的是我很好调试,两题都是几乎一遍过的. 介绍树链剖分前,先确保已经学会以下基本技巧: ...

  3. Luogu P2146 软件包管理器(树链剖分+线段树)

    题意 给定\(n\)个软件包,每个软件包都有一个依赖软件包,安装一个软件包必须安装他的依赖软件包,卸载一个软件包必须先卸载所有依赖于它的软件包.给定\(m\)此操作,每次一个操作\(install/u ...

  4. 【Luogu】P2146软件包管理器(树链剖分)

    题目链接 上午跟rqy学了一道超难的概率题,准备颓一会,于是水了这么一道水题. 话说这题真的是模板啊.数据范围正好,描述特别贴近(都不给你绕弯子的),连图都给你画出来,就差题目描述加一句“树链剖分模板 ...

  5. NOI2015 软件包管理器(树链剖分+线段树)

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

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

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

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

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

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

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

  9. AC日记——软件包管理器 洛谷 P2416

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

随机推荐

  1. 快照方式备份MySQL数据库及举例

    快照方式备份MySQL数据库及举例 作者: 红豆殺 日期: 2011 年 03 月 17 日发表评论7条评论查看评论   一.创建逻辑卷 依照如下连接的文档创建一个逻辑卷 http://www.178 ...

  2. 面试题常考&必考之--http访问一个页面的全流程(Tcp/IP协议)

    分析:-http访问一个页面的全流程,也就好比我们在地址栏输入地址,然后点击回车进行访问 该面试题的主要考点是:计算机网络的TCP/IP协议栈 描述图片:首先应用层提交http请求,传到传输层后由,T ...

  3. CMD命令行二

    图形化用户界面 calc control mmc notepad regedit (start命令也有用) dir | findstr /i "for" 忽略大小写查找 finds ...

  4. PHP入门培训教程PHP程序员要掌握哪些技术

    总有那么一群人,学个半吊子就急着找工作,面试题做不出来,吹牛都吹不来所以你只能低工资.PHP程序员要掌握哪些技术?那么兄弟连PHP培训 就来小结一下. 面试前请参考:(前三阶段完成80%在北京月薪5k ...

  5. rk3328设备树学习

    一.用到的rk3328好像使用了设备树 设备树我知道的有三种文件类型,dtbs是通过指令make dtbs编译的二进制文件,供内核使用. 基于同样的软件分层设计的思想,由于一个SoC可能对应多个mac ...

  6. 在CentOS7中配置网络时常见的LSB加载失败问题

    前几天,为了给OpenNebula扩展新的主机节点,对CentOS7的网络进行了配置.本以为网络配置只需要简单修改ifcfg-eth0即可,但是在重启网络服务时却遇到了一个LSB加载失败的问题(Fai ...

  7. http协议的深刻理解

    https://www.cnblogs.com/mayite/p/9095986.html

  8. RabbitMQ幂等性概念(七)

    幂等性是什么? 我们可以借鉴数据库的乐观锁机制 比如我们执行一条更新库存的sql语句update user set count=count-1,version=version+1 where vers ...

  9. 按ECS退出全屏模式

    <!DOCTYPE html><html><meta http-equiv="Content-Type" content="text/htm ...

  10. Linux下修改mysql root密码

    1.修改MySQL的配置文件(默认为/etc/my.cnf),在[mysqld]下添加一行skip-grant-tables 2.保存配置文件后,重启MySQL服务 service mysqld re ...