E. Egor in the Republic of Dagestan(Codeforces Round #669 (Div. 2))

题目链接: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
*/

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值