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