博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
BZOJ1922:[SDOI2010]大陆争霸(最短路)
阅读量:6832 次
发布时间:2019-06-26

本文共 2566 字,大约阅读时间需要 8 分钟。

Description

在一个遥远的世界里有两个国家:位于大陆西端的杰森国和位于大陆东端的 克里斯国。两个国家的人民分别信仰两个对立的神:杰森国信仰象征黑暗和毁灭 的神曾·布拉泽,而克里斯国信仰象征光明和永恒的神斯普林·布拉泽。 幻想历 8012年 1月,杰森国正式宣布曾·布拉泽是他们唯一信仰的神,同 时开始迫害在杰森国的信仰斯普林·布拉泽的克里斯国教徒。 幻想历 8012年 3月2日,位于杰森国东部小镇神谕镇的克里斯国教徒发动 起义。 幻想历 8012年 3月7日,神谕镇的起义被杰森国大军以残酷手段镇压。 幻想历 8012年 3月8日,克里斯国对杰森国宣战。由数十万大军组成的克 里斯军团开至两国边境,与杰森军团对峙。 幻想历 8012年 4月,克里斯军团攻破杰森军团防线进入神谕镇,该镇幸存 的克里斯国教徒得到解放。 战争随后进入胶着状态,旷日持久。战况惨烈,一时间枪林弹雨,硝烟弥漫, 民不聊生。 幻想历 8012年 5月12日深夜,斯普林·布拉泽降下神谕:“Trust me, earn eternal life.”克里斯军团士气大增。作为克里斯军团的主帅,你决定利用这一机 会发动奇袭,一举击败杰森国。具体地说,杰森国有 N 个城市,由 M条单向道 路连接。神谕镇是城市 1而杰森国的首都是城市 N。你只需摧毁位于杰森国首都 的曾·布拉泽大神殿,杰森国的信仰,军队还有一切就都会土崩瓦解,灰飞烟灭。 为了尽量减小己方的消耗,你决定使用自爆机器人完成这一任务。唯一的困 难是,杰森国的一部分城市有结界保护,不破坏掉结界就无法进入城市。而每个 城市的结界都是由分布在其他城市中的一些结界发生器维持的,如果想进入某个 城市,你就必须破坏掉维持这个城市结界的所有结界发生器。 现在你有无限多的自爆机器人,一旦进入了某个城市,自爆机器人可以瞬间 引爆,破坏一个目标(结界发生器,或是杰森国大神殿),当然机器人本身也会 一起被破坏。你需要知道:摧毁杰森国所需的最短时间。

Input

第一行两个正整数 N, M。 接下来 M行,每行三个正整数 ui, vi, wi,表示有一条从城市ui到城市 vi的单 向道路,自爆机器人通过这条道路需要 wi的时间。 之后 N 行,每行描述一个城市。首先是一个正整数 li,维持这个城市结界所 使用的结界发生器数目。之后li个1~N 之间的城市编号,表示每个结界发生器的 位置。如果 Li = 0,则说明该城市没有结界保护,保证L1 = 0 。

Output

仅包含一个正整数 ,击败杰森国所需的最短时间。

Sample Input

6 6
1 2 1
1 4 3
2 3 1
2 5 2
4 6 2
5 3 2
0
0
0
1 3
0
2 3 5

Sample Output

5

HINT

对于 20%的数据,满足 N≤15,M≤50;

对于 50%的数据,满足 N≤500,M≤6,000;
对于 100%的数据,满足 N≤3,000,M≤70,000,1≤wi≤108
输入数据保证一定有解,且不会存在维持某个城市结界的结界发生器在这个
城市内部。
连接两个城市的道路可能不止一条, 也可能存在一个城市自己到自己的道路。

Solution

计算两个数组$dis1[x]$和$dis2[x]$,分别表示到这个点的最短路和这个点的最早可进入时间,所以$max(dis1[x],dis2[x])$就是这个点的实际进入时间。

每次从堆里面取出来一个点,然后去更新$dis1$,更新完$dis1$了再去更新他保护节点的$dis2$,然后这个点保护的节点的度数都减$1$。如果入度为$0$了就入队。

Code

1 #include
2 #include
3 #include
4 #include
5 #include
6 #define N (3009) 7 using namespace std; 8 9 struct Node10 {11 int num,dis;12 bool operator < (const Node &a) const {
return dis>a.dis;}13 };14 struct Edge{
int to,next,len;}edge[N<<5];15 int n,m,u,v,l,x;16 int dis1[N],dis2[N],vis[N],Ind[N];17 int head[N],num_edge;18 vector
vec[N];19 priority_queue
q;20 21 void add(int u,int v,int l)22 {23 if (u==v) return;24 edge[++num_edge].to=v;25 edge[num_edge].next=head[u];26 edge[num_edge].len=l;27 head[u]=num_edge;28 }29 30 void Dijkstra()31 {32 memset(dis1,0x7f,sizeof(dis1));33 dis1[1]=0; q.push((Node){ 1,0});34 while (!q.empty())35 {36 int x=q.top().num; q.pop();37 if (vis[x]) continue; vis[x]=1;38 int maxn=max(dis1[x],dis2[x]);39 for (int i=head[x]; i; i=edge[i].next)40 if (maxn+edge[i].len

转载于:https://www.cnblogs.com/refun/p/10163396.html

你可能感兴趣的文章
vue的生命周期(又称钩子函数)----以及vue1.0版本与vue2.0版本生命周期的不同
查看>>
java数组复制===clone()
查看>>
《Thinking in Java》学习笔记(六)
查看>>
测试博客功能
查看>>
leetcode:Number of 1 Bits
查看>>
在SUBLIME TEXT中安装SUBLIMELINTER进行JS&CSS代码校验
查看>>
RaspberryPi 使用4G无线上网卡
查看>>
Struts2 系列问题一
查看>>
fastdfs-nginx扩展模块源码分析
查看>>
第十七条:要么为继承而设计,并提供文档说明,要么就禁止继承
查看>>
LSTM-航班人数预测
查看>>
编译器学习(二)非确定有穷自动机的整理
查看>>
专题2.永恒的黄金(中高风险)
查看>>
Linux 典型应用之Mysql
查看>>
架构设计之策略模式
查看>>
理解距(数学)
查看>>
web 开发之js---js 实现网页中播放wav的一种方法(flash播放器)
查看>>
openwrt下部署adbyby去广告大师 免luci 带自启动,自动开启透明代理
查看>>
[.Net 多线程处理系列专题七——对多线程的补充
查看>>
shell code one
查看>>