#\(\mathcal{\color{red}{Description}}\)

\(Link\)

一道\(zz\)的树剖题\(qwq\)。

#\(\mathcal{\color{red}{Solution}}\)

我天\(NOI\)怎么可能有这种水题啊……

很显然的就是 线段树区间染色 + 统计根到每个节点的距离 + 子树和 + 修改子树 ……

纯纯的板子啊!

\(emmmmm\)于是这道题就完了~

#include <cstdio>
#include <vector>
#include <iostream>
#define il inline
#define MAXN 200010
#define ls(x) (x << 1)
#define rs(x) (x << 1 | 1)
#define mid ((l + r) >> 1) using namespace std ;
char in[30] ;
vector <int> sons[MAXN] ; int Id[MAXN], tot ;
int dis[MAXN << 1], s[MAXN << 1], tag[MAXN << 1], i ;
int N, M, In, fa[MAXN], dep[MAXN], Top[MAXN], sub[MAXN], hs[MAXN] ; inline int qrr(){
int k = 0 ; char c = getchar() ;
while (c < '0' || c > '9') c = getchar() ;
while (c <= '9' && c >= '0') k = (k << 1) + (k << 3) + c - 48, c = getchar() ;
return k ;
}
void _dfs1(int deep, int now, int f){
dep[now] = deep, fa[now] = f ; sub[now] = 1 ;
int _hs = -1 ;
for (int k = 0 ; k < sons[now].size() ; ++ k){
_dfs1(deep + 1, sons[now][k], now) ;
sub[now] += sub[sons[now][k]] ;
if(sub[sons[now][k]] > _hs)
_hs = sub[sons[now][k]], hs[now] = sons[now][k] ;
}
}
void _dfs2(int now, int Tp){
Top[now] = Tp, Id[now] = ++ tot ;
if (!hs[now]) return ;
_dfs2(hs[now], Tp) ;
for (int k = 0 ; k < sons[now].size() ; ++ k){
if (sons[now][k] == hs[now]) continue ;
_dfs2(sons[now][k], sons[now][k]) ;
}
}
void build(int rt, int l, int r){
if (l == r) {dis[rt] = 1 ; return ;}
build(ls(rt), l, mid) ;
build(rs(rt), mid + 1, r) ;
dis[rt] = dis[ls(rt)] + dis[rs(rt)] ;
}
il void p_d(int rt, int l, int r){
if(tag[rt] != -1){
s[ls(rt)] = (mid - l + 1) * tag[rt] ;
s[rs(rt)] = (r - mid) * tag[rt] ;
tag[ls(rt)] = tag[rs(rt)] = tag[rt] ;
tag[rt] = -1 ;
}
}
int query_D(int rt, int l, int r, int ql, int qr){
int res = 0 ;
if (ql <= l && r <= qr){res += dis[rt] ; return res ;}
if (ql <= mid) res += query_D(ls(rt), l, mid, ql, qr) ;
if (qr > mid) res += query_D(rs(rt), mid + 1, r, ql, qr) ;
return res ;
}
int query_C(int rt, int l, int r, int ql, int qr){
int res = 0 ;
if (ql <= l && r <= qr){res += s[rt] ; return res ;}
p_d(rt, l, r) ;
if (ql <= mid) res += query_C(ls(rt), l, mid, ql, qr) ;
if (qr > mid) res += query_C(rs(rt), mid + 1, r, ql, qr) ;
return res ;
}
void update_C(int rt, int l, int r, int ul, int ur, int k){
if (ul <= l && r <= ur){
s[rt] = (r - l + 1) * k ;
tag[rt] = k ;
return ;
}
p_d(rt, l, r) ;
if (ul <= mid) update_C(ls(rt), l, mid, ul, ur, k) ;
if (ur > mid) update_C(rs(rt), mid + 1, r, ul, ur, k) ;
s[rt] = s[rs(rt)] + s[ls(rt)] ;
}
inline void qv(int now){
int ans = 0, dif = 0 ;
while (Top[now] != Top[1]){
ans += query_D(1, 1, N, Id[Top[now]], Id[now]) ;
dif += query_C(1, 1, N, Id[Top[now]], Id[now]) ;
now = fa[Top[now]] ;
}
ans += query_D(1, 1, N, 1, Id[now]) ;
dif += query_C(1, 1, N, 1, Id[now]) ;
printf("%d\n", ans - dif) ;
}
inline void uv(int now){
while (Top[now] != Top[1]){
update_C(1, 1, N, Id[Top[now]], Id[now], 1) ;
now = fa[Top[now]] ;
}
update_C(1, 1, N, 1, Id[now], 1) ;
}
int main(){
//freopen("manager.in", "r", stdin) ;
//freopen("manager.out", "w", stdout) ;
N = qrr() ;
fill (tag + 1, tag + (MAXN << 1) + 1, -1) ;
for (i = 2; i <= N ;i ++){
fa[i] = qrr() + 1,
sons[fa[i]].push_back(i) ;
} M = qrr() ;
_dfs1(1, 1, 0), _dfs2(1, 1), build(1, 1, N) ;
for (i = 1; i <= M ; i ++){
scanf("%s", in) ;
if (in[0] == 'u'){
In = qrr() + 1 ;
printf("%d\n", query_C(1, 1, N, Id[In], Id[In] + sub[In] - 1)) ;
update_C(1, 1, N, Id[In], Id[In]+ sub[In] - 1, 0) ;
continue ;
}
In = qrr() + 1 ;
qv(In), uv(In) ;
}
}

番外:以下是三次评测:

最快的是\(qread + printf + O2\),其次是\(qread + printf\),最慢的是\(qread + cout\)……

[题解向] Luogu2146[NOI2015]软件包管理器的更多相关文章

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

    题目大意 ​ 给你一棵树, 求一点到根的路径上有多少个未标记点并全标记, 和询问一个点的子树内有多少已标记点和撤销标记 解题方法 1: install 操作 ​ 这个操作是求一点到根的路径上有多少个未 ...

  2. luogu2146 [NOI2015]软件包管理器

    安装就把根节点到它全设为 1 删除就把以它为根的子树全设为 0 记得标记初始化为-1,因为标记是 0 的情况也是要处理的. #include <iostream> #include < ...

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

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

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

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

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

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

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

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

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

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

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

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

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

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

随机推荐

  1. 【转】2019年7月份,阿里最新Java高频面试真题汇总

    技术一面(23问)技术二面(3大块)JAVA开发技术面试中可能问到的问题(17问)JAVA方向技术考察点(33快)项目实战(7大块)必会知识(48点)面试小技巧注意事项1. 阿里技术一面 Java I ...

  2. .NET / C# HTTP中的GET和PSOT

    需要引入using System.IO;using System.Net; public string GETs(string URL) { //创建httpWebRequest对象 HttpWebR ...

  3. Linux中Crontab的使用

    一.安装依赖 yum install cronie 二.添加 Crontab crontab -e 三.查看crontab内容 crontab -l 其中常见的一些内容 例子: # 每月的最后1天 0 ...

  4. 【JVM】CMS垃圾回收器

    一.简介 Concurrent Mark Sweep,是一种以获取最短回收停顿时间为目标的收集器,尤其重视服务的响应速度. CMS是老年代垃圾回收器,基于标记-清除算法实现.新生代默认使用ParNew ...

  5. uni-app悬浮框模板

    1. uni-app悬浮框模板 1.1. 目标 模仿饿了吗app的悬浮框效果,即上移过程中,中间的某个组件框到顶部后不再上移,呈类似置顶的效果 1.2. 问题 中间遇到fixed固定组件导致flex失 ...

  6. Cheat Engine 人造指针

    打开游戏 查看内存区域 查看游戏当前使用的内存区域 下面这一段是游戏当前使用的内存区域,选择一片可以读写的内存区域 跳转到这片内存 查看是否有空余内存可以使用 使用空闲内存 我们选择0075DFD0开 ...

  7. Unity手游汉化笔记②:使用UABE替换TTF字体

    总的笔记:https://www.cnblogs.com/guobaoxu/p/12055930.html 目录 一.分析 二.思路 三.具体实践 四.总结 Unity版本:2018.4.5f1 工具 ...

  8. SAP CDS redirect view支持写操作吗,一个实验来验证

    According to this wiki, write back on CDS view is not supported: And also it is defined in ABAP help ...

  9. 共享变量与python测试库

    共享变量(Variables) 1.变量表: 导入模式: 1.套件文件,不需要导入 2.资源文件 Resource  xxx.robot 2.变量文件 导入模式: 1.Variables xx.py ...

  10. prometheus学习系列九: Prometheus AlertManager使用

    在Prometheus的报警系统中,是分为2个部分的, 规则是配置是在prometheus中的, prometheus组件完成报警推送给alertmanager的, alertmanager然后管理这 ...