BZOJ4003[JLOI2015]城池攻占——可并堆
题目描述
小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池。
输入
第 1 行包含两个正整数 n;m,表示城池的数量和骑士的数量。
输出
输出 n + m 行,每行包含一个非负整数。其中前 n 行分别表示在城池 1 到 n 牺牲的骑士
样例输入
50 20 10 10 30
1 1 2
2 0 5
2 0 -10
1 0 10
20 2
10 3
40 4
20 4
35 5
样例输出
2
0
0
0
1
1
3
1
1
提示
对于 100% 的数据,1 <= n;m <= 300000; 1 <= fi<i; 1 <= ci <= n; -10^18 <= hi,vi,si <= 10^18;ai等于1或者2;当 ai =1 时,vi > 0;保证任何时候骑士战斗力值的绝对值不超过 10^18。
可以发现修改操作只有加一个数和乘一个正数,也就是说两个数的大小关系不会在同时修改后改变。那么我们可以对树上的每个节点维护一个可并堆(小根堆)然后自下而上合并上去。对于树上的一个节点,先将这个点的可并堆与子树的可并堆依次合并,然后判断堆顶是否大于该点的占领权值,如果小于就删除堆顶直到堆顶大于等于该点权值。之后将这个点的堆打上类似线段树的懒惰标记即可。注意乘法标记和加法标记的顺序问题。
#include<set>
#include<map>
#include<queue>
#include<stack>
#include<cmath>
#include<cstdio>
#include<bitset>
#include<vector>
#include<cstring>
#include<iostream>
#include<algorithm>
#define ll long long
using namespace std;
ll h[300010];
int f[300010];
int a[300010];
ll v[300010];
ll s[300010];
int c[300010];
int dis[300010];
int ls[300010];
int rs[300010];
int head[300010];
int to[300010];
int nex[300010];
ll num[300010];
ll sum[300010];
int root[300010];
int res[300010];
int dep[300010];
int ans[300010];
int tot;
int n,m;
void add(int x,int y)
{
nex[++tot]=head[x];
head[x]=tot;
to[tot]=y;
}
void add(int rt,ll k,ll b)
{
if(!rt)
{
return ;
}
s[rt]*=k,s[rt]+=b;
sum[rt]*=k,num[rt]*=k,num[rt]+=b;
}
void pushdown(int rt)
{
add(ls[rt],sum[rt],num[rt]);
add(rs[rt],sum[rt],num[rt]);
sum[rt]=1,num[rt]=0;
}
int merge(int x,int y)
{
if(!x||!y)
{
return x+y;
}
pushdown(x);
pushdown(y);
if(s[x]>s[y])
{
swap(x,y);
}
rs[x]=merge(rs[x],y);
if(dis[ls[x]]<dis[rs[x]])
{
swap(ls[x],rs[x]);
}
dis[x]=dis[rs[x]]+1;
return x;
}
void dfs(int x)
{
for(int i=head[x];i;i=nex[i])
{
dep[to[i]]=dep[x]+1;
dfs(to[i]);
root[x]=merge(root[x],root[to[i]]);
}
if(!root[x])
{
return ;
}
while(s[root[x]]<h[x]&&root[x])
{
pushdown(root[x]);
ans[root[x]]=dep[c[root[x]]]-dep[x];
res[x]++;
root[x]=merge(ls[root[x]],rs[root[x]]);
}
if(a[x])
{
add(root[x],v[x],0);
}
else
{
add(root[x],1,v[x]);
}
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
{
scanf("%lld",&h[i]);
}
for(int i=2;i<=n;i++)
{
scanf("%d%d%lld",&f[i],&a[i],&v[i]);
add(f[i],i);
}
for(int i=1;i<=m;i++)
{
scanf("%lld%d",&s[i],&c[i]);
sum[i]=1ll;
root[c[i]]=merge(root[c[i]],i);
}
dep[1]=1;
dfs(1);
while(root[1])
{
ans[root[1]]=dep[c[root[1]]];
root[1]=merge(ls[root[1]],rs[root[1]]);
}
for(int i=1;i<=n;i++)
{
printf("%d\n",res[i]);
}
for(int i=1;i<=m;i++)
{
printf("%d\n",ans[i]);
}
}
BZOJ4003[JLOI2015]城池攻占——可并堆的更多相关文章
- 【BZOJ4003】[JLOI2015]城池攻占 可并堆
[BZOJ4003][JLOI2015]城池攻占 Description 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池. 这 n 个城池用 1 到 n 的整数表示.除 1 号 ...
- [bzoj4003][JLOI2015]城池攻占_左偏树
城池攻占 bzoj-4003 JLOI-2015 题目大意:一颗n个节点的有根数,m个有初始战斗力的骑士都站在节点上.每一个节点有一个standard,如果这个骑士的战斗力超过了这个门槛,他就会根据城 ...
- BZOJ4003 [JLOI2015]城池攻占 左偏树 可并堆
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ4003 题意概括 题意有点复杂,直接放原题了. 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑 ...
- [BZOJ4003][JLOI2015]城池攻占(左偏树)
这题有多种做法,一种是倍增预处理出每个点往上走2^i步最少需要的初始战斗力,一种是裸的启发式合并带标记splay. 每个点合并能攻占其儿子的所有骑士,删去所有无法攻占这个城市的骑士并记录答案. 注意到 ...
- BZOJ4003 [JLOI2015]城池攻占
这题有两种做法来着... 第一种就是一开始想到的比较不靠谱,不过貌似可以过掉: 看从$1$号节点开始到$p$号节点最大需要的体力,记录单调上升的体力,询问的时候二分跳着走就可以了 不过精度问题还有可能 ...
- BZOJ4003 JLOI2015城池攻占
用左偏树模拟攻占的过程,维护最小值,最多入和出m次,每次log复杂度. #include<bits/stdc++.h> using namespace std; ; typedef lon ...
- BZOJ_4003_[JLOI2015]城池攻占_可并堆
BZOJ_4003_[JLOI2015]城池攻占_可并堆 Description 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池. 这 n 个城池用 1 到 n 的整数表示.除 ...
- [洛谷P3261] [JLOI2015]城池攻占(左偏树)
不得不说,这道题目是真的难,真不愧它的“省选/NOI-”的紫色大火题!!! 花了我晚自习前半节课看题解,写代码,又花了我半节晚自习调代码,真的心态爆炸.基本上改得和题解完全一样了我才过了这道题!真的烦 ...
- BZOJ-4003:城池攻占(可并堆+lazy标记)
小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池. 这 n 个城池用 到 n 的整数表示.除 号城池外,城池 i 会受到另一座城池 fi 的管辖, 其中 fi <i.也就是 ...
随机推荐
- CentOS 7 源码编译安装 Redis
1.下载源码并解压 wget http://download.redis.io/releases/redis-4.0.10.tar.gz tar -xzf redis-4.0.10.tar.gz cd ...
- 使用docker Registry快速搭建私有镜像仓库
当我们执行docker pull xxx的时候,docker默认是从registry.docker.com这个地址上去查找我们所需要的镜像文件,然后执行下载操作.这类的镜像仓库就是docker默认的公 ...
- 面试 9:Java 玩转冒泡排序
面试 9:用 Java 实现冒泡排序 南尘的朋友们,新的一周好,原本打算继续讲链表考点算法的,这里姑且是卡一段.虽然在我们 Android 开发中,很少涉及到排序算法,因为基本官方都帮我们封装好了,但 ...
- 最新版XCoder 的使用方法
1.项目中,新建一个类库.名字随意,我取名:XCoder 2.右键 > 管理nuget程序包:搜索 XCode 并安装 3.在项目中新建:data.project.xml 的xml文件,并写入数 ...
- Typescript 发布到npm
https://blog.csdn.net/yiershan1314/article/details/79999726 https://cloud.tencent.com/developer/arti ...
- Spring MVC+ Spring + Mybatis从零开始搭建一个精美且实用的管理后台
点击进入<SSM搭建精美实用的管理系统>达人课页面 SSM 框架即 SpringMVC+Spring+Mybatis,相信各位朋友在投递简历时已直观感受到它的重要性,JavaWeb 相关工 ...
- immutability因React官方出镜之使用总结分享!
作者:首席填坑官∙苏南 公众号:honeyBadger8,群:912594095,本文原创,著作权归作者所有,转载请注明原链接及出处. 引言 之前项目中遇到数据拷贝.引用之间数据层级嵌套过深,拷贝的值 ...
- Python_每日习题_0006_斐波那契数列
程序设计: 斐波那契数列(Fibonacci sequence),从1,1开始,后面的每一项等于前面两项之和. 图方便就递归实现,图性能就用循环. # for 循环 target = int(inpu ...
- 2017湘潭大学邀请赛H题(树的直径)
链接:https://www.icpc.camp/contests/4mYguiUR8k0GKE H. Highway The input contains zero or more test cas ...
- Karen and Coffee CodeForces - 816B (差分数组+预处理前缀和)
To stay woke and attentive during classes, Karen needs some coffee! Karen, a coffee aficionado, want ...