bzoj 4003: 城池攻占 左偏树
题目大意
题解
一开始看漏条件了
题目保证当占领城池可以使攻击力乘上\(v_i\)时,一定有\(v_i>0\)
上面这句话很重要(或者说去掉这句话就可以出成一道新题)
我们考虑在每个节点上都维护一个最小堆
每次移动时我们让在堆中的所有骑士一起移动
这样我们知道:如果堆顶的骑士可以占领成功
那么这个堆里剩下的所有骑士都一定能够占领成功
如果堆顶元素无法占领我们就删去这个堆顶的元素,这样我们最多删除n次
最坏情况是\(O(nlogn)\)的,可以承受
所以我们还需要这个堆支持快速合并,任意一个可并堆即可
于是我敲了一棵左偏树上去...打标记打到手软...第一次在堆上打标记
这个标记的正确性就由上面加粗的话保证了
这句话保证了:打标记后树的结构不会改变
调了1h才发现是merge没有return...
神奇的windows系统还让我过了样例和对拍,linux虚拟机下直接爆炸
Code
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
template<typename T>inline void read(T &x){
x=0;char ch;bool flag = false;
while(ch=getchar(),ch<'!');if(ch == '-') ch=getchar(),flag = true;
while(x=10*x+ch-'0',ch=getchar(),ch>'!');if(flag) x=-x;
}
inline int cat_max(const int &a,const int &b){return a>b ? a:b;}
inline int cat_min(const int &a,const int &b){return a<b ? a:b;}
const int maxn = 300010;
struct Node{
Node *ch[2];
ll val,add,mul;
int cnt,tag,deg,id;
}*null;
Node mem[maxn<<1],*li;
Node *root[maxn];
inline void init(){
li = mem;null = li++;
null->ch[0] = null->ch[1] = null;
null->val = null->add = null->cnt
= null->tag = null->mul = 0;
}
inline Node* newNode(int id,ll val){
Node *p = li++;p->val = val;
p->ch[0] = p->ch[1] = null;
p->cnt = p->tag = p->deg = p->add = 0;
p->mul = 1;p->id = id;return p;
}
inline void pushdown(Node *p){
if(p == null) return;
if(p->ch[0] != null){
Node *x = p->ch[0];
if(p->mul != 1){x->mul *= p->mul;x->add *= p->mul;x->val *= p->mul;}
if(p->add != 0){x->add += p->add;x->val += p->add;}
if(p->tag != 0){x->tag += p->tag;x->cnt += p->tag;}
}
if(p->ch[1] != null){
Node *x = p->ch[1];
if(p->mul != 1){x->mul *= p->mul;x->add *= p->mul;x->val *= p->mul;}
if(p->add != 0){x->add += p->add;x->val += p->add;}
if(p->tag != 0){x->tag += p->tag;x->cnt += p->tag;}
}p->mul = 1;p->tag = p->add = 0;
}
Node *merge(Node *x,Node *y){
//printf("%d %d\n",x,y);
if(x == null) return y;
if(y == null) return x;
if(x->val > y->val) swap(x,y);
pushdown(x);
x->ch[1] = merge(x->ch[1],y);
if(x->ch[0]->deg < x->ch[1]->deg)
swap(x->ch[0],x->ch[1]);
x->deg = x->ch[1]->deg + 1;
return x;
}
int ans1[maxn],ans2[maxn];
inline void pop_less(int i,ll k,int p){
while(1){
if(root[i] == null) break;
if(root[i]->val >= k) break;
pushdown(root[i]);
ans2[root[i]->id] = root[i]->cnt;
//printf("%d died on %d with killed %d\n",root[i]->id,p,root[i]->cnt);
//printf("%lld <-> %lld\n",root[i]->val,k);
root[i] = merge(root[i]->ch[0],root[i]->ch[1]);
ans1[p]++;
}
}
ll h[maxn],v[maxn],s[maxn];
int c[maxn],fa[maxn],a[maxn];
inline void update(Node *p,int i){
//printf("update");
p->tag++;p->cnt++;
if(a[i] == 0){
p->add += v[i];
p->val += v[i];
}else{
p->mul *= v[i];
p->add *= v[i];
p->val *= v[i];
}
}
int main(){
init();
int n,m;read(n);read(m);
for(int i=1;i<=n;++i) read(h[i]),root[i] = null;
//printf("%d\n",root[2]);
for(int i=2;i<=n;++i){
read(fa[i]);read(a[i]);read(v[i]);
}
for(int i=1;i<=m;++i){
read(s[i]);read(c[i]);
if(s[i] < h[c[i]]){
ans1[c[i]]++;
ans2[i] = 0;
}else{
Node *p = newNode(i,s[i]);
update(p,c[i]);
root[c[i]] = merge(root[c[i]],p);
}
}
for(int i=n;i>=2;--i){
pop_less(i,h[fa[i]],fa[i]);
update(root[i],fa[i]);
root[fa[i]] = merge(root[i],root[fa[i]]);
}
while(root[1] != null){
pushdown(root[1]);
ans2[root[1]->id] = root[1]->cnt;
//printf("%d win with killed %d\n",root[1]->id,root[1]->cnt);
root[1] = merge(root[1]->ch[0],root[1]->ch[1]);
}
for(int i=1;i<=n;++i) printf("%d\n",ans1[i]);
for(int i=1;i<=m;++i) printf("%d\n",ans2[i]);
getchar();getchar();
return 0;
}
bzoj 4003: 城池攻占 左偏树的更多相关文章
- BZOJ 4003: [JLOI2015]城池攻占 左偏树 可并堆
https://www.lydsy.com/JudgeOnline/problem.php?id=4003 感觉就是……普通的堆啊(暴论),因为这个堆是通过递归往右堆里加一个新堆或者新节点的,所以要始 ...
- BZOJ 4003 / Luogu P3261 [JLOI2015]城池攻占 (左偏树)
左偏树裸题,在树上合并儿子传上来的堆,然后小于当前结点防御值的就pop掉,pop的时候统计答案. 修改的话就像平衡树一样打懒标记就行了. 具体见代码 CODE #include<bits/std ...
- [JLOI2015]城池攻占 左偏树
题目描述 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池.这 n 个城池用 1 到 n 的整数表示.除 1 号城池外,城池 i 会受到另一座城池 fi 的管辖,其中 fi &l ...
- [luogu3261 JLOI2015] 城池攻占 (左偏树+标记)
传送门 Description 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池.这 n 个城池用 1 到 n 的整数表示.除 1 号城池外,城池 i 会受到另一座城池 fi 的 ...
- [洛谷P3261] [JLOI2015]城池攻占(左偏树)
不得不说,这道题目是真的难,真不愧它的“省选/NOI-”的紫色大火题!!! 花了我晚自习前半节课看题解,写代码,又花了我半节晚自习调代码,真的心态爆炸.基本上改得和题解完全一样了我才过了这道题!真的烦 ...
- bzoj 4003 [JLOI2015]城池攻占 —— 左偏树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4003 其实蛮简单的,首先一个城市只会被其子树中的骑士经过,启发我们 dfs 序用可并堆合并子 ...
- BZOJ4003 [JLOI2015]城池攻占 左偏树 可并堆
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ4003 题意概括 题意有点复杂,直接放原题了. 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑 ...
- [BZOJ4003][JLOI2015]城池攻占(左偏树)
这题有多种做法,一种是倍增预处理出每个点往上走2^i步最少需要的初始战斗力,一种是裸的启发式合并带标记splay. 每个点合并能攻占其儿子的所有骑士,删去所有无法攻占这个城市的骑士并记录答案. 注意到 ...
- BZOJ 1455 罗马游戏 ——左偏树
[题目分析] 左偏树的模板题目,大概就是尽量维护树的深度保持平衡,以及尽可能的快速合并的一种堆. 感觉和启发式合并基本相同. 其实并没有快很多. 本人的左偏树代码自带大常数,借鉴请慎重 [代码] #i ...
随机推荐
- Chrome + Python 抓取动态网页内容
用Python实现常规的静态网页抓取时,往往是用urllib2来获取整个HTML页面,然后从HTML文件中逐字查找对应的关键字.如下所示: import urllib2 url="http: ...
- python学习(十四)面向对象
Python中的面向对象,先写类,会生成类对象,类对象然后创建对象,对象就可以拿来用了. Python支持多重继承. class语句创建类对象,并将其赋值给变量名. class语句内的赋值语句会创建类 ...
- python升级或者其他原因把yum搞坏了
第一个命令查询出来,原本是安装的啥版本 rpm -qa | grep python- | grep 2.6 然后执行下一个命令,就可以安装原本的python版本了,注意链接要换成你对应的那个版本 rp ...
- Linux模块机制浅析_转
Linux模块机制浅析 转自:http://www.cnblogs.com/fanzhidongyzby/p/3730131.htmlLinux允许用户通过插入模块,实现干预内核的目的.一直以来,对l ...
- 【WPF学习笔记】之如何点登录按钮时判断用户名密码进行登录:动画系列之(二)
...... 承接动画系列之(一)的代码: 再添加登录按钮代码进行登录,验证用户名和密码在数据库是否正确. 直接上代码: using System; using System.Collections. ...
- 基于imgAreaSelect的用户图像截取
前言:想到用户资料中一般有个图像自我截取的部分,为什么要截取呢,因为好看了.so,经过我各种百度,各种参考,终于打工搞成了,写下纪念纪念,让以后拿来就用也好. 一:想前端ui这东西,我就懒得说话了,哎 ...
- C#游戏开发高速新手教程Unity5.5教程
C#游戏开发高速新手教程Unity5.5教程 试读文档下载地址:http://pan.baidu.com/s/1slwBHoD C#是微软公布的高级程序设计语言.这门语言和C语言一样,已经成为了大学计 ...
- 转载 ----Android学习笔记 - 蓝牙篇 (Bluetooth)
1.什么是蓝牙 Bluetooth是目前使用的最广泛的无线通讯协议之一 主要针对短距离设备通讯(10米) 常用于连接耳机.鼠标和移动通讯设备等 2.发现周围蓝牙设备 BluetoothAd ...
- UnicodeEncodeError: ‘ascii’ codec can’t encode characters in position xxx ordinal not in range(12
python在安装时,默认的编码是ascii,当程序中出现非ascii编码时,python的处理常常会报这样的错UnicodeDecodeError: 'ascii' codec can't deco ...
- python 基础 1.5 数据类型(二)--列表
一.python 数据类型序列---列表 1.列表是可变型的数据类型.列表里边的元素是可变的,可以增加,可以删除. 2.列表(list)是处理一组有序项目的数据结构,即可以在列表中存储一个序列的项 ...