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


    题目描述

    小铭铭最近获得了一副新的桌游,游戏中需要用 m 个骑士攻占 n 个城池。

    这 n 个城池用 1 到 n 的整数表示。除 1 号城池外,城池 i 会受到另一座城池 fi 的管辖,
    其中 fi <i。也就是说,所有城池构成了一棵有根树。这 m 个骑士用 1 到 m 的整数表示,其
    中第 i 个骑士的初始战斗力为 si,第一个攻击的城池为 ci。
    每个城池有一个防御值 hi,如果一个骑士的战斗力大于等于城池的生命值,那么骑士就可
    以占领这座城池;否则占领失败,骑士将在这座城池牺牲。占领一个城池以后,骑士的战斗力
    将发生变化,然后继续攻击管辖这座城池的城池,直到占领 1 号城池,或牺牲为止。
    除 1 号城池外,每个城池 i 会给出一个战斗力变化参数 ai;vi。若 ai =0,攻占城池 i 以后骑士战斗力会增加 vi;若 ai =1,攻占城池 i 以后,战斗力会乘以 vi。注意每个骑士是单独计算的。也就是说一个骑士攻击一座城池,不管结果如何,均不会影响其他骑士攻击这座城池的结果。
    现在的问题是,对于每个城池,输出有多少个骑士在这里牺牲;对于每个骑士,输出他攻占的城池数量。

    输入

    第 1 行包含两个正整数 n;m,表示城池的数量和骑士的数量。

    第 2 行包含 n 个整数,其中第 i 个数为 hi,表示城池 i 的防御值。
    第 3 到 n +1 行,每行包含三个整数。其中第 i +1 行的三个数为 fi;ai;vi,分别表示管辖
    这座城池的城池编号和两个战斗力变化参数。
    第 n +2 到 n + m +1 行,每行包含两个整数。其中第 n + i 行的两个数为 si;ci,分别表
    示初始战斗力和第一个攻击的城池。

    输出

     输出 n + m 行,每行包含一个非负整数。其中前 n 行分别表示在城池 1 到 n 牺牲的骑士

    数量,后 m 行分别表示骑士 1 到 m 攻占的城池数量。

    样例输入

    5 5
    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
    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
    ",res[i]);
    	}
    	for(int i=1;i<=m;i++)
    	{
    		printf("%d
    ",ans[i]);
    	}
    }
  • 相关阅读:
    Omi新成员omi-router正式发布
    Omi架构与React Fiber
    Omi框架Store体系的前世今生
    Omi v1.0震撼发布
    omi-cli新版发布-升级webpack2和支持sass生成组件局部CSS
    Omi应用md2site-0.5.0发布-支持动态markdown拉取解析
    Omi应用md2site发布-markdown转网站利器
    AlloyTouch之无限循环select插件
    Omi教程-插件体系
    AlloyTouch之select选择插件
  • 原文地址:https://www.cnblogs.com/Khada-Jhin/p/10652909.html
Copyright © 2020-2023  润新知