BZOJ 4003 / Luogu P3261 [JLOI2015]城池攻占 (左偏树)
左偏树裸题,在树上合并儿子传上来的堆,然后小于当前结点防御值的就pop掉,pop的时候统计答案.
修改的话就像平衡树一样打懒标记就行了.
具体见代码
CODE
#include<bits/stdc++.h>
using namespace std;
char cb[1<<15],*cs=cb,*ct=cb;
#define getc() (cs==ct&&(ct=(cs=cb)+fread(cb,1,1<<15,stdin),cs==ct)?0:*cs++)
template<class T>inline void read(T &res) {
char ch; int flg = 1; for(;!isdigit(ch=getchar());)if(ch=='-')flg=-flg;
for(res=ch-'0';isdigit(ch=getchar());res=res*10+ch-'0'); res*=flg;
}
typedef long long LL;
const int MAXN = 300005;
const int MAXM = 300005;
#define lc t[x].ls
#define rc t[x].rs
struct node {
int ls, rs, d;
LL v, add_tag, mul_tag;
}t[MAXN];
int n, m, cnt, ans1[MAXN], ans2[MAXM];
int dep[MAXN], fir[MAXN], fa[MAXN], c[MAXM];
LL v[MAXN], a[MAXN], h[MAXN];
struct edge { int to, nxt; }e[MAXN];
inline void add(int u, int v) { e[++cnt] = (edge) { v, fir[u] }, fir[u] = cnt; }
vector<int> here[MAXN];
inline void upd(int x) {
if(t[lc].d < t[rc].d) swap(lc, rc);
t[x].d = t[rc].d + 1;
}
inline void mt(int x) {
if(t[x].mul_tag != 1) { //题目中满足只会乘正数,所以能够用堆维护
if(lc)
t[lc].mul_tag *= t[x].mul_tag,
t[lc].add_tag *= t[x].mul_tag,
t[lc].v *= t[x].mul_tag;
if(rc)
t[rc].mul_tag *= t[x].mul_tag,
t[rc].add_tag *= t[x].mul_tag,
t[rc].v *= t[x].mul_tag;
t[x].mul_tag = 1;
}
if(t[x].add_tag) {
if(lc)
t[lc].add_tag += t[x].add_tag,
t[lc].v += t[x].add_tag;
if(rc)
t[rc].add_tag += t[x].add_tag,
t[rc].v += t[x].add_tag;
t[x].add_tag = 0;
}
}
int merge(int x, int y){
if(!x || !y) return x + y;
mt(x), mt(y);
if(t[x].v > t[y].v) swap(x, y);
t[x].rs = merge(t[x].rs, y);
upd(x);
return x;
}
int pop(int x) { mt(x);
int l = t[x].ls, r = t[x].rs;
t[x].ls = t[x].rs = t[x].add_tag = 0; t[x].mul_tag = 1;
return merge(l, r);
}
int dfs(int x) {
int rt = 0;
while(!here[x].empty())
rt = merge(rt, here[x].back()), here[x].pop_back();
for(int i = fir[x]; i; i = e[i].nxt)
dep[e[i].to] = dep[x] + 1, rt = merge(rt, dfs(e[i].to));
while(rt && t[rt].v < h[x])
++ans1[x], ans2[rt] = dep[c[rt]]-dep[x], rt = pop(rt);
if(rt && x > 1) {
if(a[x] == 0) t[rt].v += v[x], t[rt].add_tag += v[x];
else t[rt].v *= v[x], t[rt].add_tag *= v[x], t[rt].mul_tag *= v[x];
}
return rt;
}
int main() {
read(n), read(m); t[0].d = -1;
for(int i = 1; i <= n; ++i) read(h[i]);
for(int i = 2; i <= n; ++i) read(fa[i]), read(a[i]), read(v[i]), add(fa[i], i);
for(int i = 1; i <= m; ++i) read(t[i].v), read(c[i]), here[c[i]].push_back(i), t[i].mul_tag = 1;
int root = dfs(1);
while(root) ans2[root] = dep[c[root]] + 1, root = pop(root); //不要忘了有打通关了的骑士
for(int i = 1; i <= n; ++i) printf("%d\n", ans1[i]);
for(int i = 1; i <= m; ++i) printf("%d\n", ans2[i]);
}
BZOJ 4003 / Luogu P3261 [JLOI2015]城池攻占 (左偏树)的更多相关文章
- [洛谷P3261] [JLOI2015]城池攻占(左偏树)
不得不说,这道题目是真的难,真不愧它的“省选/NOI-”的紫色大火题!!! 花了我晚自习前半节课看题解,写代码,又花了我半节晚自习调代码,真的心态爆炸.基本上改得和题解完全一样了我才过了这道题!真的烦 ...
- BZOJ 4003: [JLOI2015]城池攻占 左偏树 可并堆
https://www.lydsy.com/JudgeOnline/problem.php?id=4003 感觉就是……普通的堆啊(暴论),因为这个堆是通过递归往右堆里加一个新堆或者新节点的,所以要始 ...
- [JLOI2015]城池攻占 左偏树
题目描述 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池.这 n 个城池用 1 到 n 的整数表示.除 1 号城池外,城池 i 会受到另一座城池 fi 的管辖,其中 fi &l ...
- [luogu3261 JLOI2015] 城池攻占 (左偏树+标记)
传送门 Description 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池.这 n 个城池用 1 到 n 的整数表示.除 1 号城池外,城池 i 会受到另一座城池 fi 的 ...
- bzoj 4003 [JLOI2015]城池攻占 —— 左偏树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4003 其实蛮简单的,首先一个城市只会被其子树中的骑士经过,启发我们 dfs 序用可并堆合并子 ...
- BZOJ4003 [JLOI2015]城池攻占 左偏树 可并堆
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ4003 题意概括 题意有点复杂,直接放原题了. 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑 ...
- [BZOJ4003][JLOI2015]城池攻占(左偏树)
这题有多种做法,一种是倍增预处理出每个点往上走2^i步最少需要的初始战斗力,一种是裸的启发式合并带标记splay. 每个点合并能攻占其儿子的所有骑士,删去所有无法攻占这个城市的骑士并记录答案. 注意到 ...
- bzoj 4003: 城池攻占 左偏树
题目大意 http://www.lydsy.com/JudgeOnline/problem.php?id=4003 题解 一开始看漏条件了 题目保证当占领城池可以使攻击力乘上\(v_i\)时,一定有\ ...
- P3261 [JLOI2015]城池攻占 (左偏树+标记下传)
左偏树还是满足堆的性质,节点距离就是离最近的外节点(无左或者右儿子 或者二者都没有)的距离,左偏性质就是一个节点左儿子的距离不小于右儿子,由此得:节点距离等于右儿子的距离+1. 本题就是对于每个节点 ...
随机推荐
- ULB
https://docs.ucloud.cn/network/ulb/intro/whatisulb 创建ulb ulb里创建vserver, 设置监听端口 配置后端转发节点(服务节点) 在服务节点上 ...
- GitHub访问及git工具克隆慢问题解决
一.查询合适的DNS服务器 1.站长工具DNS查询服务器 github.com github.global.ssl.fastly.net 2.选择TTL值最小的记录,记录下对应服务器IP地址 例如 g ...
- X86逆向13:向程序中插入Dll
本章我们将学习Dll的注入技巧,我们将把一个动态链接库永久的插入到目标程序中,让程序在运行后直接执行这个Dll文件,这一章的内容也可以看作是第八课的加强篇,第八课中我们向程序中插入了一个弹窗,有木有发 ...
- Ubuntu系统开机后不能正常使用——问题解决记录
1.开机后桌面内容没了,搜狗输入法不能使用了,终端不能打开了 问题原因:上次关机前为了解决解压文件中文乱码问题,在/etc/profile末尾加了如下两行:(但事实上如下两行根本不能解决中文乱码问题) ...
- WCF寄宿windows服务一
如果只是寄宿单个wcf服务,方法很简单,步骤:1.创建好一个windows服务.关于windows服务内容见:http://www.cnblogs.com/zhaow/p/7866916.html2. ...
- 这周末又参加班里同学生日party,同学父母包场2小时花费大约1000美金左右。
今天班上Claire的生日,邀请了几个小朋友去pump it up.特别特别开心,因为她父母选的时间特别好晚上6-8点小孩子玩疯了以后吃的特别多.
- 03 Go语言特性
一.基本注意事项 1.转义字符 \t 一个制表符,代表一次tab \n 换行符 \\ 转义代表 \ \" 转义代表 " \r 一个回车,从当前行的最前面开始输出,会覆盖以前的内容, ...
- 前后端分离-模拟数据之RAP2快速入门
是啥? RAP是一个可视化接口管理工具 通过分析接口结构,动态生成模拟数据,校验真实接口正确性, 围绕接口定义,通过一系列自动化工具提升我们的协作效率.我们的口号:提高效率,回家吃晚饭! 可视化编辑, ...
- 林大妈的CSS知识清单(二)可见格式化模型(内含margin塌陷与浮动闭合的解决方案)
盒模型.浮动和定位是CSS中最重要的三个概念.它们共同决定了一个元素在页面中以怎样的形式进行排布与显示. 一.盒模型 1. 定义 盒模型是CSS的核心概念.一个页面中,所有的元素(不管他最终显示是圆形 ...
- element-ui select
1. 组合 label <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> & ...