Dijkstra【P2446】 [SDOI2010]大陆争霸

Background

在一个遥远的世界里有两个国家:位于大陆西端的杰森国和位于大陆东端的克里斯国。两个国家的人民分别信仰两个对立的神:杰森国信仰象征黑暗和毁灭的神曾·布拉泽,而克里斯国信仰象征光明和永恒的神斯普林·布拉泽。

幻想历8012年1月,杰森国正式宣布曾·布拉泽是他们唯一信仰的神,同时开始迫害在杰森国的信仰斯普林·布拉泽的克里斯国教徒。

幻想历8012年3月2日,位于杰森国东部小镇神谕镇的克里斯国教徒发动起义。

幻想历8012年3月7日,神谕镇的起义被杰森国大军以残酷手段镇压。

幻想历8012年3月8日,克里斯国对杰森国宣战。由数十万大军组成的克里斯军团开至两国边境,与杰森军团对峙。

幻想历8012年4月,克里斯军团攻破杰森军团防线进入神谕镇,该镇幸存的克里斯国教徒得到解放。

战争随后进入胶着状态,旷日持久。战况惨烈,一时间枪林弹雨,硝烟弥漫,民不聊生。

Description

幻想历8012年5月12日深夜,斯普林·布拉泽降下神谕:“Trust me, earn eternal life.”克里斯军团士气大增。作为克里斯军团的主帅,你决定利用这一机会发动奇袭,一举击败杰森国。具体地说,杰森国有N个城市,由M条单向道路连接。神谕镇是城市1而杰森国的首都是城市N。你只需摧毁位于杰森国首都的曾·布拉泽大神殿,杰森国的信仰,军队还有一切就都会土崩瓦解,灰飞烟灭。

为了尽量减小己方的消耗,你决定使用自爆机器人完成这一任务。唯一的困难是,杰森国的一部分城市有结界保护,不破坏掉结界就无法进入城市。而每个城市的结界都是由分布在其他城市中的一些结界发生器维持的,如果想进入某个城市,你就必须破坏掉维持这个城市结界的所有结界发生器。

现在你有无限多的自爆机器人,一旦进入了某个城市,自爆机器人可以瞬间引爆,破坏一个目标(结界发生器,或是杰森国大神殿),当然机器人本身也会一起被破坏。你需要知道:摧毁杰森国所需的最短时间。

Input

输入文件的landcraft.in的第一行两个正整数N, M。

接下来M行,每行三个正整数ui, vi, wi,表示有一条从城市ui到城市vi的单向道路,自爆机器人通过这条道路需要wi的时间。

之后N行,每行描述一个城市。首先是一个正整数li,维持这个城市结界所使用的结界发生器数目。之后li个1~N之间的城市编号,表示每个结界发生器的位置。如果li = 0,则说明该城市没有结界保护,保证l1 = 0 。

Output

输出文件landcraft.out仅包含一个正整数 ,击败杰森国所需的最短时间。

最短路问题,结果因为数组开小改了好久???

记录\(dis[x]\)代表到达\(x\)的最短时间。

记录\(real[x]\)代表到达\(x\)的实际时间。

对于每一个点,我们去更新其相连节点的时候要用\(max(dis[x],real[x])\)去更新。

然后注意建立结界保护的边的时候建\(li\)到\(i\)的有向边。

因为我搞不清所以直接建双向边

然后最后输出答案输出\(max(dis[n],real[n])\)即可。

代码

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<queue>
#define R register

using namespace std;

const int gz=5e4+8;

inline void in(R int &x)
{
    R int f=1;x=0;char s=getchar();
    while(!isdigit(s)){if(s=='-')f=-1;s=getchar();}
    while(isdigit(s)){x=x*10+s-'0';s=getchar();}
    x*=f;
}

int head[gz],tot,pr[gz],cnt,hd[gz];

struct cod{int u,v,w;}edge[gz<<1],e[gz<<1];

inline void add(R int x,R int y,R int z)
{
    edge[++tot].u=head[x];
    edge[tot].v=y;
    edge[tot].w=z;
    head[x]=tot;
}

inline void ado(R int x,R int y)
{
    e[++cnt].u=pr[x];
    e[cnt].v=y;
    pr[x]=cnt;
}

int dis[gz],real[gz],n,m;

bool vis[gz];

struct hop
{
    int u,d;
    bool operator <(const hop&a)const
    {
        return d>a.d;
    };
};

inline void dij()
{
    for(R int i=1;i<=n;i++)dis[i]=2147483644;
    priority_queue<hop>q;dis[1]=real[1]=0;
    q.push((hop){1,0});
    while(!q.empty())
    {
        R int u=q.top().u;q.pop();
        if(vis[u])continue;
        vis[u]=true;
        R int now=max(dis[u],real[u]);
        for(R int i=head[u];i;i=edge[i].u)
        {
            if(dis[edge[i].v]>now+edge[i].w)
            {
                dis[edge[i].v]=now+edge[i].w;
                if(hd[edge[i].v]==0)
                    q.push((hop){edge[i].v,max(dis[edge[i].v],real[edge[i].v])});
            }
        }
        for(R int i=pr[u];i;i=e[i].u)
        {
            hd[e[i].v]--;
            real[e[i].v]=max(real[e[i].v],now);
            if(hd[e[i].v]==0)
                q.push((hop){e[i].v,max(real[e[i].v],dis[e[i].v])});
        }
    }
    printf("%d\n",max(real[n],dis[n]));
}

int main()
{
    in(n),in(m);
    for(R int i=1,x,y,z;i<=m;i++)
    {
        in(x),in(y),in(z);
        if(x==y)continue;
        add(x,y,z);
    }
    for(R int i=1,x;i<=n;i++)
    {
        in(x);hd[i]=x;
        for(R int fk;x;x--)
            in(fk),ado(fk,i),ado(i,fk);
    }
    dij();
}

原文地址:https://www.cnblogs.com/-guz/p/9975215.html

时间: 2024-10-16 04:26:01

Dijkstra【P2446】 [SDOI2010]大陆争霸的相关文章

P2446 [SDOI2010]大陆争霸(有限制的最短刘)

题目描述 幻想历8012年5月12日深夜,斯普林·布拉泽降下神谕:“Trust me, earn eternal life.”克里斯军团士气大增.作为克里斯军团的主帅,你决定利用这一机会发动奇袭,一举击败杰森国.具体地说,杰森国有N个城市,由M条单向道路连接.神谕镇是城市1而杰森国的首都是城市N.你只需摧毁位于杰森国首都的曾·布拉泽大神殿,杰森国的信仰,军队还有一切就都会土崩瓦解,灰飞烟灭. 为了尽量减小己方的消耗,你决定使用自爆机器人完成这一任务.唯一的困难是,杰森国的一部分城市有结界保护,不

AC日记——[SDOI2010]大陆争霸 洛谷 P3690

[SDOI2010]大陆争霸 思路: dijkstra模板: 代码: #include <bits/stdc++.h> using namespace std; #define maxn 3005 #define ll long long #define maxm 70002<<2 #define INF 1e13 struct NodeType { ll id,dis; bool operator<(const NodeType pos)const { return dis

BZOJ 1922: [Sdoi2010]大陆争霸

二次联通门 : BZOJ 1922: [Sdoi2010]大陆争霸 /* BZOJ 1922: [Sdoi2010]大陆争霸 最短路思路题 带限制的转移 _link[x] 记录的是当前城市有多少城市保护 记录两个距离 一个是到当前点的最短路 一个是最早能够进入当前点时间 每次枚举当前点的必到点 更新dis_2 若都炸完后加入堆中 则答案就是 max (dis_1[N], dis_2[N]); */ #include <cstring> #include <cstdio> #incl

1922: [Sdoi2010]大陆争霸|dijkstra

最短路大法好.Orz黄学长 设d1[x],d2[x]为城市x的到达时间,可进入时间 max(d1[x],d2[x])为真实的进入时间 d[x]记录城市x被多少个城市保护 每次堆中取出一个真实进入时间最小的城市 更新它所通往的城市的d1,保护城市的d2 保护城市的d–1 若d=0,则可入堆 复杂度(n+m)log2n 黄学长写的比较清楚了,似乎很多人都用了矩阵存图的样子,怪不得一遍卡进第一页 #include<iostream> #include<algorithm> #includ

bzoj 1922 [Sdoi2010]大陆争霸(最短路变形)

Description 在一个遥远的世界里有两个国家:位于大陆西端的杰森国和位于大陆东端的 克里斯国.两个国家的人民分别信仰两个对立的神:杰森国信仰象征黑暗和毁灭 的神曾·布拉泽,而克里斯国信仰象征光明和永恒的神斯普林·布拉泽. 幻想历 8012年 1月,杰森国正式宣布曾·布拉泽是他们唯一信仰的神,同 时开始迫害在杰森国的信仰斯普林·布拉泽的克里斯国教徒. 幻想历 8012年 3月2日,位于杰森国东部小镇神谕镇的克里斯国教徒发动 起义. 幻想历 8012年 3月7日,神谕镇的起义被杰森国大军以残

BZOJ1922 [Sdoi2010]大陆争霸 【最短路】

题目 在一个遥远的世界里有两个国家:位于大陆西端的杰森国和位于大陆东端的 克里斯国.两个国家的人民分别信仰两个对立的神:杰森国信仰象征黑暗和毁灭 的神曾·布拉泽,而克里斯国信仰象征光明和永恒的神斯普林·布拉泽. 幻想历 8012年 1月,杰森国正式宣布曾·布拉泽是他们唯一信仰的神,同 时开始迫害在杰森国的信仰斯普林·布拉泽的克里斯国教徒. 幻想历 8012年 3月2日,位于杰森国东部小镇神谕镇的克里斯国教徒发动 起义. 幻想历 8012年 3月7日,神谕镇的起义被杰森国大军以残酷手段镇压. 幻想

BZOJ1922 SDOI2010 大陆争霸 最短路

题意:给定一个图,图中有保护关系(u,v)表示到v之前必须先到一次u,求从1到N的最短路 题解: 定义d1[i]为直接到达i的最短距离,这个的更新和普通的Dijkstra一样 定义d2[i]为解除i的所有保护的最短距离(不一定要在i结束),这个更新起来很简单,每经过一个节点就将其所控制的城市的发生器数全部--,没有发生器的城市直接用当前的距离更新即可. 最后答案显然就是max(d1[N],d2[N]) #include <queue> #include <vector> #incl

bzoj 1922: [Sdoi2010]大陆争霸 带限制最短路

题意:有n个点和m条道路.有一些点必须经过了某些点后才能到达,问从起点到达终点的最短时间. 分析:带限制的最短路. 设d1[x]为可以进入点x的时间,d2[x]为到达点x的时间,s[x]为点x被多少个点保护. 一开始以为d1和d2是有关联的,也就是要通过d1来推出某些d2,然后想破脑袋都想不出(其实也没那么夸张啦).后来看了题解才发现这两个可以分开求,然后进入点x的实际时间为max(d1[x],d2[x]). 然后跑dijkstra,每次找到一个d2最小且s[x]=0的点,然后更新与这个点相连的

[SDOI2010]大陆争霸

嘟嘟嘟 首先可以知道,对于在哪个时候攻占一个城市,应该是他的最短到达时间和最早进入时间的最大值(max(d1[i], d2[i])). 最短到达时间:就是朴素的最短路d1[i]. 最早进入时间:设所有到达有他的结界发生器的城市为j,那么应该是在所有最短时间中取max,作为d2[i]. 于是就可以用dijkstra写了. 每一次成功更新节点 i 的d1时,就更新所有 i 能控制的节点 j 的d2. 我们还要记录每一个节点的入度,代表控制它的城市有几个,然后如果他的d2被更新了一次,就减1,若果为0