[AHOI2014/JSOI2014]骑士游戏(SPFA认识)

传送门

好题!考察了对于SPFA的深刻理解。

对于每个怪,我们要么花费魔攻代价,要么花费普攻代价+消灭其儿子的代价。

看似像个一直递归下去的做法,仿佛可以用dp?

但是图中可能会存在环,就会有后效性。

假如我们用一个队列保存当前还需要消灭的怪。

我们每次提出队首,要么魔攻,要么普攻然后又压入一堆新出现的怪。直到队为空。

感觉这个过程有点像SPFA。

SPFA是对于每次那些还可能松弛更新的边进行松弛,直到无法操作(队空)就保证了最短路。

对于这道题,我们好像也可以每次看能不能再更新消灭怪的最小花费,直到不能再更新。

即使有正环也不会影响最短路。

如果连边,显然不是很好操作。不如考虑直接模拟SPFA的操作过程。

dis[i]存的不是起点1到它的最短路,我们定义dis[i]为消灭i及其产物的最小花费

我们考虑从某一个怪被魔攻(一定会有一些怪被魔攻,不然产生怪无止境),其对于父亲的dis会做出贡献。

或者说一个怪不一定是魔攻,但消灭其及其产物的最小代价是dis[i],这个dis[i]会对父亲的dis做出贡献。

把父亲的魔攻代价和普攻代价+消灭其儿子的代价相比进行更新操作。

如果父亲被更新了,说明父亲的父亲也还有可能更新,于是把父亲的父亲再次入队。

所以把dis[i]一开始就设为这个怪被魔攻的代价。

我们保存一个点的儿子和它可以更新的父亲。

跑一遍SPFA,最后输出dis[1]即可。

#include<bits/stdc++.h>
#define LL long long
#define N 200003
#define M 1000003
using namespace std;
LL read()
{
    LL x=0,f=1;char s=getchar();
    while(s<‘0‘||s>‘9‘){if(s==‘-‘)f=-1;s=getchar();}
    while(s>=‘0‘&&s<=‘9‘){x=x*10+s-‘0‘;s=getchar();}
    return x*f;
}
struct EDEG{
    int nextt,to;
}w[M];
vector<int>fa[N];
LL s[N],dis[N];
int vis[N],head[N];
int tot=0,n;
void add(int a,int b)
{
    tot++;
    w[tot].nextt=head[a];
    w[tot].to=b;
    head[a]=tot;
}
queue<int>q;
void SPFA()
{
    while(!q.empty())q.pop();
    for(int i=1;i<=n;++i)vis[i]=1,q.push(i);
    while(!q.empty())
    {
        int x=q.front();q.pop();
        vis[x]=0;
        LL tmp=s[x];
        for(int i=head[x];i;i=w[i].nextt)
        {
            int v=w[i].to;
            tmp+=dis[v];
        }
        if(tmp>=dis[x])continue;
        dis[x]=tmp;
        for(int i=0;i<fa[x].size();++i)
        {
            int f=fa[x][i];
            if(vis[f])continue;
            vis[f]=1;q.push(f);
        }
    }
    printf("%lld\n",dis[1]);
}
int main()
{
    n=read();
    for(int i=1;i<=n;++i)
    {
        s[i]=read(),dis[i]=read();
        int num=read();
        while(num--)
        {
            int x=read();
            add(i,x);fa[x].push_back(i);
        }
    }
    SPFA();
} 

原文地址:https://www.cnblogs.com/yyys-/p/11697472.html

时间: 2024-08-03 15:04:12

[AHOI2014/JSOI2014]骑士游戏(SPFA认识)的相关文章

[AHOI2014/JSOI2014]骑士游戏(SPFA的本质)

题意 消灭一个魔物有两种方法,消耗\(a_i\)代价将它彻底消灭,或者消耗\(b_i\)代价将它变成多个其他魔物,求消灭1号魔物的最小代价 思路 SPFA:一个点的最短路径被更新后它的儿子节点才可能被更新,于是将该点入队,这样做减少了没有更新的无用枚举 对于这道题,设\(dis_i\)表示一个\(i\)魔物被彻底消灭的代价,(显然最多不超过\(a_i\)),类似上面的思路,不过这里将可能被更新的点入队(而不是上面的被更新过的点),取出队头的一个点,更新它的\(dis\)值,如果更新了就将所有可以

BZOJ3875: [Ahoi2014&amp;Jsoi2014]骑士游戏

Description [故事背景] 长期的宅男生活中,JYY又挖掘出了一款RPG游戏.在这个游戏中JYY会 扮演一个英勇的骑士,用他手中的长剑去杀死入侵村庄的怪兽. [问题描述] 在这个游戏中,JYY一共有两种攻击方式,一种是普通攻击,一种是法术攻 击.两种攻击方式都会消耗JYY一些体力.采用普通攻击进攻怪兽并不能把怪兽彻底杀死,怪兽的尸体可以变出其他一些新的怪兽,注意一个怪兽可能经过若干次普通攻击后变回一个或更多同样的怪兽:而采用法术攻击则可以彻底将一个怪兽杀死.当然了,一般来说,相比普通攻

【BZOJ3875】【Ahoi2014】骑士游戏 SPFA处理有后效性动规

广告: #include <stdio.h> int main() { puts("转载请注明出处[vmurder]谢谢"); puts("网址:blog.csdn.net/vmurder/article/details/44040735"); } 题解: 首先一个点可以分裂成多个新点,这样就有了图上动规的基础. 即f[i]表示i点被消灭的最小代价,它可以由分裂出的点们更新. 但是这个东西有后效性,所以我们用SPFA来处理它. spfa处理后效性动规 我

[AHOI2014/JSOI2014]骑士游戏

题目 思博贪心题写了一个半小时没救了,我也没看出这是一个\(spfa\)来啊 设\(dp_i\)表示彻底干掉第\(i\)只怪物的最小花费,一个非常显然的事情,就是对于\(k_i\)值最小的怪物满足\(dp_i=k_i\) 非常好理解,反正到最后都要干掉这个怪物,何必再把它干成别的怪物 于是我们按照\(k_i\)的值先排序一下,另外维护一个小根堆 如果堆里没有点或者堆顶的\(dp\)值比当前的\(k\)要大,我们直接令当前当前\(k_i\)值最小的点\(i\)的\(dp_i=k_i\),之后遍历所

luogu P4042 [AHOI2014/JSOI2014]骑士游戏

这道题首先想到Dp.. 不妨设 f[i] 表示杀死怪兽 i 需要的最少体力, 那么有 f[i] = min( k[i] ,  s[i] + ∑ f [ri] ) , 这还是很好想的.. 但是没有办法转移啊,因为可能有环,高斯消元??,不存在的.. 然后就开始搞最短路了,但怎样建图呢,一般的跑最短路都是有起点,有终点,有边权,但这道题除了知道起点是 1 外,其他啥也没有.. 于是我们就需要深入理解SPFA的本质,所以说这是一道很好的思维题... SPFA是去不断的找最优解(松弛)..... 那么我

BZOJ 3875 Ahoi2014 骑士游戏 SPFA

题目大意:给定n个怪物,每个怪物可以用魔法直接干掉,或者用物理攻击使其分裂为一些其他怪物,求杀掉1号怪物的最小花销 令f[i]为杀死i号怪物的最小花销,则f[i]=min(k[i],s[i]+Σf[j]) 其中j为i用物理攻击后可以分裂为的怪物 但是直接DP有后效性,因此我们用SPFA来跑这个DP即可 注意如果每次更新一个点之后都重新计算花销会T掉 改成减掉花销的差值就好了 具体写法去看代码吧- - #include <cstdio> #include <cstring> #inc

「AHOI2014/JSOI2014」骑士游戏

「AHOI2014/JSOI2014」骑士游戏 传送门 考虑 \(\text{DP}\). 设 \(dp_i\) 表示灭种(雾)一只编号为 \(i\) 的怪物的代价. 那么转移显然是: \[dp_i = \min(K_i, S_i + \sum_{j = 1}^{R_i} dp_{v_j})\] 但是我们会发现这个东西是有后效性的... 所以我们会想要用建图然后跑一个最短路什么的来搞... 于是我们观察到上面那个 \(\text{DP}\) 式子中,\(dp_i\) 如果用后面那一项来转移,显然

BZOJ 3875 Ahoi2014 骑士游戏

3875: [Ahoi2014]骑士游戏 Time Limit: 30 Sec  Memory Limit: 256 MB Description [故事背景] 长期的宅男生活中,JYY又挖掘出了一款RPG游戏.在这个游戏中JYY会 扮演一个英勇的骑士,用他手中的长剑去杀死入侵村庄的怪兽. [问题描述] 在这个游戏中,JYY一共有两种攻击方式,一种是普通攻击,一种是法术攻击.两种攻击方式都会消耗JYY一些体力.采用普通攻击进攻怪兽并不能把怪兽彻底杀死,怪兽的尸体可以变出其他一些新的怪兽,注意一个

bzoj3876 [Ahoi2014&amp;Jsoi2014]支线剧情

3876: [Ahoi2014&Jsoi2014]支线剧情 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 2180  Solved: 1300[Submit][Status][Discuss] Description [故事背景] 宅男JYY非常喜欢玩RPG游戏,比如仙剑,轩辕剑等等.不过JYY喜欢的并不是战斗场景,而是类似电视剧一般的充满恩怨情仇的剧情.这些游戏往往 都有很多的支线剧情,现在JYY想花费最少的时间看完所有的支线剧情. [问题描述