题意

有一棵树\(n\)个节点,每个节点有一个防御值,以及两个属性,表示一个骑士占领该节点后攻击值是加还是乘,有\(m\)个骑士,有初始位置和初始攻击值,如果攻击值大于该节点的防御值,就能占领该节点,然后更新攻击值,走到父节点,如果攻击值小于防御值,骑士就会死在该节点。

问每个骑士能占领多少个节点,以及每个节点分别有多少个骑士死在那里。

分析

  • 第一个问题,考虑对每一个节点,如果我们能知道所有能到达该节点的骑士以及他们的攻击力,显然攻击力小于该节点防御值的就是死在这个节点的骑士。
  • 这部分骑士分为两部分,第一部分是初始位置就在这个节点的,第二部分是从下面上来的,这部分可以用dfs来求出,然后考虑用可并堆来维护这些骑士的信息。
  • 显然将以该节点为初始位置的骑士和dfs后回溯上来的骑士对应的可并堆进行合并,然后将攻击力小于防御值的骑士去掉,维护大根堆,显然这些骑士也不可能再对上面的节点有贡献。
  • 第二个问题,由于骑士走的肯定是树上的一个单向路径,所以只需要记录初始位置的深度和死亡位置的深度即可。

代码

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=3e5+50;
struct Edge{
int v,next;
}e[N],ct[N];
int cnt1,cnt2,head1[N],head2[N];
void init(){
cnt1=0;
cnt2=0;
memset(head1,-1,sizeof(head1));
memset(head2,-1,sizeof(head1));
}
void add(int u,int v,bool tr){
if(tr){
e[cnt1]=Edge{v,head1[u]};
head1[u]=cnt1++;
}else{
ct[cnt2]=Edge{v,head2[u]};
head2[u]=cnt2++;
}
}
int n,m,fa,fi[N],sis[N],k[N],ls[N],rs[N],dis[N],dep[N];
ll f[N],ai[N],vi[N],g[N],ad[N],mu[N];
//对a子树计算标记
void fun(int a,ll add,ll mul){
if(a){
g[a]*=mul;
g[a]+=add;
ad[a]*=mul;
ad[a]+=add;
mu[a]*=mul;
}
}
void pushdown(int a){
fun(ls[a],ad[a],mu[a]);
fun(rs[a],ad[a],mu[a]);
ad[a]=0;
mu[a]=1;
}
int merge(int a,int b){
if(!a || !b){
return a+b;
}
pushdown(a);
pushdown(b);
if(g[a]>g[b]){
swap(a,b);
}
rs[a]=merge(rs[a],b);
if(dis[ls[a]]<dis[rs[a]]){
swap(ls[a],rs[a]);
}
dis[a]=dis[rs[a]]+1;
return a;
}
int pop(int a){
pushdown(a);
return merge(ls[a],rs[a]);
}
int dfs(int u,int d){
//因为是小根堆,这里是a=0,如果是大根堆,a=u ???
int a=0;
dep[u]=d;
//合并在这个城池开始的所有骑士
for(int i=head2[u];i!=-1;i=ct[i].next){
int v=ct[i].v;
a=merge(a,v);
}
//合并能从下面上来到这个城池的骑士
for(int i=head1[u];i!=-1;i=e[i].next){
int v=e[i].v;
a=merge(a,dfs(v,d+1));
}
//攻击力不够的骑士死在这个城池,记录死的位置,通过深度可知占领的城池数
while(a && g[a]<f[u]){
k[a]=u;
sis[u]++;
a=pop(a);
}
//更新攻击力,回溯到上一层城池进行攻击
if(ai[u]){
fun(a,0,vi[u]);
}else{
fun(a,vi[u],1);
}
return a;
}
int main(){
// freopen("in.txt","r",stdin);
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++){
scanf("%lld",&f[i]);
}
init();
for(int i=2;i<=n;i++){
scanf("%d%lld%lld",&fa,&ai[i],&vi[i]);
add(fa,i,true);
}
for(int i=1;i<=m;i++){
scanf("%lld%d",&g[i],&fi[i]);
add(fi[i],i,false);
}
dfs(1,1);
for(int i=1;i<=n;i++){
printf("%d\n",sis[i]);
}
for(int i=1;i<=m;i++){
printf("%d\n",dep[fi[i]]-dep[k[i]]);
}
return 0;
}

luoguP3261_[JLOI2015]城池攻占的更多相关文章

  1. BZOJ_4003_[JLOI2015]城池攻占_可并堆

    BZOJ_4003_[JLOI2015]城池攻占_可并堆 Description 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池. 这 n 个城池用 1 到 n 的整数表示.除 ...

  2. 【BZOJ4003】[JLOI2015]城池攻占 可并堆

    [BZOJ4003][JLOI2015]城池攻占 Description 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池. 这 n 个城池用 1 到 n 的整数表示.除 1 号 ...

  3. [bzoj4003][JLOI2015]城池攻占_左偏树

    城池攻占 bzoj-4003 JLOI-2015 题目大意:一颗n个节点的有根数,m个有初始战斗力的骑士都站在节点上.每一个节点有一个standard,如果这个骑士的战斗力超过了这个门槛,他就会根据城 ...

  4. [洛谷P3261] [JLOI2015]城池攻占(左偏树)

    不得不说,这道题目是真的难,真不愧它的“省选/NOI-”的紫色大火题!!! 花了我晚自习前半节课看题解,写代码,又花了我半节晚自习调代码,真的心态爆炸.基本上改得和题解完全一样了我才过了这道题!真的烦 ...

  5. [JLOI2015]城池攻占

    题目描述 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池.这 n 个城池用 1 到 n 的整数表示.除 1 号城池外,城池 i 会受到另一座城池 fi 的管辖,其中 fi &l ...

  6. [JLOI2015]城池攻占 左偏树

    题目描述 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池.这 n 个城池用 1 到 n 的整数表示.除 1 号城池外,城池 i 会受到另一座城池 fi 的管辖,其中 fi &l ...

  7. BZOJ4003[JLOI2015]城池攻占——可并堆

    题目描述 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池. 这 n 个城池用 1 到 n 的整数表示.除 1 号城池外,城池 i 会受到另一座城池 fi 的管辖, 其中 fi ...

  8. BZOJ4003 [JLOI2015]城池攻占 左偏树 可并堆

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ4003 题意概括 题意有点复杂,直接放原题了. 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑 ...

  9. 【左偏树】【P3261】 [JLOI2015]城池攻占

    Description 小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池.这 n 个城池用 1 到 n 的整数表示.除 1 号城池外,城池 i 会受到另一座城池 fi 的管辖,其 ...

随机推荐

  1. CF1213D Equalizing by Division

    easy version hard version 问题分析 直接从hard version入手.不难发现从一个数\(x\)能得到的数个数是\(O(\log x)\)的.这样总共有\(O(n\log ...

  2. JavaWeb_Ajax通过JQuery和原生js异步传输数据

    菜鸟教程 传送门 AJAX 优点:在不重新加载整个页面的情况下,可以与服务器交换数据并更新部分网页内容 XMLHttpRequest 对象 传送门 (一) [JQuery]定时发送ajax请求 (二) ...

  3. spring框架中AOP思想与各种配置详解

    Spring中提供两种AOP支持:   1.基于代理的经典AOP   2.Aspectj注解配置AOP    首先我们先了解什么是AOP,AOP(Aspect Oriented Programming ...

  4. ES6学习一--ES5

    JSON对象方法 1.将JSON字符串转换成js对象(IE7及以下不支持)(IE7及以下兼容需其他手段) JSON.parse() 2.将js对象转成JSON字符串 JSON.stringify() ...

  5. 查询Oracle表空间使用情况

    ,),'990.99')||'%' "使用比(%)",F.TOTAL_BYTES "空闲空间(M)",F.MAX_BYTES "最大块(M)" ...

  6. DeepFaceLab错误:DLL Load failed 找不到指定模块!

    这个错误不知道多少人遇到了,我反正是看到过不少次了.但是一直没有花时间去研究. 今日有空帮群友远程了一下,虽然搞了一会儿,最终还是搞定了,分享一下经验. ​ 问题描述:在执行2号脚本,视频转图片的时候 ...

  7. Jmeter(七)参数化

    初识Jmeter的时候, 除了感觉安装和配置都很轻量以外, 还有一个最大的感触就是, 翻译真硬啊, 真的够够的! 和他磨合了挺长一段时间之后, 终于开悟了, 这些硬硬的翻译, 其实还是基本靠谱的, 看 ...

  8. redis源码分析之数据结构--dictionary

    本文不讲hash算法,而主要是分析redis中的dict数据结构的特性--分步rehash. 首先看下数据结构:dict代表数据字典,每个数据字典有两个哈希表dictht,哈希表采用链式存储. typ ...

  9. delphi TDbGrid 右键 PopupMenu 菜单只在有数据的地方弹出

    最近用delphi做开发,用到了DbGrid控件,想在控件上点击鼠标右键弹出菜单 关联DbGrid的 Popupmenu 倒是可以实现,但是这样的效果是不管你在哪里单击鼠标右键 只要在DBGrid里面 ...

  10. 【java基础之异常】死了都要try

    目录 1.异常 1.1 异常概念 1.2 异常体系 1.3 异常分类 1.4 异常的产生过程解析 2. 异常的处理 2.1 抛出异常throw 2.2 Objects非空判断 2.3 声明异常thro ...