[题解向] Luogu2146[NOI2015]软件包管理器
#\(\mathcal{\color{red}{Description}}\)
一道\(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]软件包管理器的更多相关文章
- 题解 P2146 【[NOI2015]软件包管理器】
题目大意 给你一棵树, 求一点到根的路径上有多少个未标记点并全标记, 和询问一个点的子树内有多少已标记点和撤销标记 解题方法 1: install 操作 这个操作是求一点到根的路径上有多少个未 ...
- luogu2146 [NOI2015]软件包管理器
安装就把根节点到它全设为 1 删除就把以它为根的子树全设为 0 记得标记初始化为-1,因为标记是 0 的情况也是要处理的. #include <iostream> #include < ...
- 【题解】NOI2015软件包管理器
[题解][P2146 NOI2015]软件包管理器 实际上就是树链剖分板子题. 对于\(install\)操作,直接查询它到\(0\)节点有多少已经安装了的,再用总数减去它. 对于\(uninstal ...
- 题解 P2146 [NOI2015]软件包管理器
P2146 [NOI2015]软件包管理器 感觉代码比其他题解更简洁qwq 树链剖分模板题 install x:将1~x的路径上的节点全部变成1(安装x需要先安装1~x) uninstall x:将x ...
- Bzoj 4196: [Noi2015]软件包管理器 树链剖分
4196: [Noi2015]软件包管理器 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 721 Solved: 419[Submit][Statu ...
- 【BZOJ4196】[Noi2015]软件包管理器 树链剖分
[Noi2015]软件包管理器 树链剖分 Description Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从 ...
- [UOJ#128][BZOJ4196][Noi2015]软件包管理器
[UOJ#128][BZOJ4196][Noi2015]软件包管理器 试题描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管 ...
- BZOJ 4196: [Noi2015]软件包管理器 [树链剖分 DFS序]
4196: [Noi2015]软件包管理器 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 1352 Solved: 780[Submit][Stat ...
- [BZOJ4196][NOI2015]软件包管理器
4196: [Noi2015]软件包管理器 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 1040 Solved: 603[Submit][Stat ...
随机推荐
- ASP.NET Core应用程序容器化、持续集成与Kubernetes集群部署(一)(转载)
本文结构 ASP.NET Core应用程序的构建 ASP.NET Core应用程序容器化所需注意的问题 应用程序的配置信息 端口侦听 ASP.NET Core的容器版本 docker镜像构建上下文(B ...
- SQL小技巧(一)拼音首字母的模糊查询
基于Microsoft SQL Server 的标量值函数fun_GetPy,借鉴其他优秀的博主文章,此处贴出源码,以及使用方法 1.打开新建查询,贴如下代码,F5 /****** Object: U ...
- 前端1-----HTML了解,内联标签(图片,超链接锚点,超链接邮箱)
前端1-----HTML了解,内联标签(图片,超链接锚点,超链接邮箱) 一丶自定制B/S # -*-coding:utf-8-*- # Author:Ds import socket IP_PORT= ...
- Prime Path POJ-3126
The ministers of the cabinet were quite upset by the message from the Chief of Security stating that ...
- webpack 里的 import, exports 实现原理
在使用 webpack 对脚本进行打包, 在开发中, 每个文件中都会使用 import 语句来导入一些功能,又会使用 export 语句导出一些功能,为了研究 import 和 export 原理,研 ...
- vue项目的各个文件作用
vue项目的各个文件作用: build:放置的是webpack配置文件,一般不动,修改了必须重启服务器才能生效 config:放置针对开发环境和线上环境的配置文件,一般不动 修改后需重启 node_m ...
- uniapp之w-picker使用采坑
1. uniapp之w-picker使用采坑 1.1. 前言 由于我是先在index页面集合了这个组件,发现该文件内容实在太多了,不好维护,所以打算把内容一个个抽成组件,在抽w-picker时,遇到了 ...
- Java 之 数学相关类 Math、BigInteger、BigDecimal
一.java.lang.Math 类 一.Math 类概述 java.lang.Math 类包含用于执行基本数学运算的方法,如指数.对数.平方根和三角函数.类似于这样的类,其所有方法均为静态方法,并且 ...
- xenserver增加新硬盘
1. 查看所有硬盘了的id # ls -l /dev/disk/by-id/ 2. 挂载 # xe sr-create type=lvm content-type=user device-config ...
- 给WEB DYNPRO 程序创建TCODE
1,创建WDA程序,这里就不介绍了,使用现成的程序:ZCRM_ME_SATISFACTION 2,SE93创建TCODE,输入事物代码ZLYTEST点击创建,选择带参数的事物代码. 3,填写事物WDY ...