题目链接:http://codeforces.com/contest/1407/problem/E
题意:给一个有向图,每条边有一个类型(0或者1),什么类型的城市就只能走什么类型的边,你需要给每个城市指定类型(0/1),使得从1到n 的最短路最长,甚至长到1无法到达n
题解:spfa
1建反向边
2dis[i][0] dis[i][1] 分别表示i为类型0/1时,从i到n的最短距离(i到n的路径上的城市类型已经指定的情况下)
3对于一条从x到y的边,边的类型为tp, dis[y][tp]=min(dis[y][tp],max(dis[x][0],dis[x][1]))
4输出每个城市的类型——只要枚举每个城市,看dis[i][0]和dis[i][1]哪个大
#include<bits/stdc++.h>
using namespace std;
const int maxn=5e5+5,inf=0x7fffffff-1;
int nxt[maxn],to[maxn],head[maxn],tp[maxn],pre[maxn][2],cl[maxn];
int n,m,cnt;
int dis[maxn][2];
bool inq[maxn];
void add(int x,int y,int k){
nxt[++cnt]=head[x]; head[x]=cnt; to[cnt]=y; tp[cnt]=k;
}
queue<int>q;
void spfa()
{
int x,y;
for(int i=1;i<=n;i++)dis[i][0]=dis[i][1]=inf;
q.push(n); dis[n][0]=dis[n][1]=0; inq[n]=1;
while(!q.empty()){
x=q.front(); q.pop(); inq[x]=0;
for(int i=head[x];i;i=nxt[i]){
y=to[i];
int tmp=max(dis[x][0],dis[x][1]);//关键
if(dis[y][tp[i]]>tmp+1){
dis[y][tp[i]]=tmp+1; pre[y][tp[i]]=x;
if(!inq[y])q.push(y),inq[y]=1;
}
}
}
}
int main()
{
int x,y,k;
scanf("%d%d",&n,&m);
for(int i=1;i<=m;i++){
scanf("%d%d%d",&x,&y,&k);
add(y,x,k);
}
spfa();
int ans=max(dis[1][0],dis[1][1]);
if(ans==inf)printf("-1\n");
else printf("%d\n",ans);
// int now=1;
// while(now!=n){
// if(dis[now][0]>dis[now][1])cl[now]=0,now=pre[now][0];
// else cl[now]=1,now=pre[now][1];
// if(now==0)break;
// }
for(int i=1;i<=n;i++){
if(dis[i][0]>dis[i][1])printf("0");
else printf("1");
}
return 0;
}
/*
4 7
1 1 0
1 3 1
3 2 0
2 1 0
3 4 1
2 4 0
2 4 1
*/