SGU 185 Two shortest 最短路删边优化内存+网络流

题目链接:点击打开链接

题意:

给定n个点m条无向边和边权(无重边)

找2条从1-n点路径不相交的最短路(2条路必须都是最短)

最先是用费用流跑,结果各种mel

然后最短路优化,,,,把图里所有不在最短路上的边删掉

然后其实跑网络流就可以了,,,开始还是mle,后来把邻接表的from去掉才过,,


#include<algorithm>
#include<queue>
#include<vector>
#include<string.h>
#include<math.h>
#include<iostream>
#include<stdio.h>
using namespace std;

#define ll int 

#define N 402
#define M 121000
#define inf 10737418
#define inf64 1152921504606846976
struct Edge{  
	ll to, cap, nex;  
}edge[M*2];//注意这个一定要够大 不然会re 还有反向弧  

ll head[N], edgenum;  
void add(ll u, ll v, ll cap, ll rw = 0){ //如果是有向边则:add(u,v,cap); 如果是无向边则:add(u,v,cap,cap); 
	Edge E = { v, cap, head[u]};  
	edge[ edgenum ] = E;  
	head[u] = edgenum ++;  

	Edge E2= { u, rw,  head[v]};  
	edge[ edgenum ] = E2;  
	head[v] = edgenum ++;  
}  
ll sign[N];  
bool BFS(ll from, ll to){  
	memset(sign, -1, sizeof(sign));  
	sign[from] = 0;  

	queue<ll>q;  
	q.push(from);  
	while( !q.empty() ){  
		int u = q.front(); q.pop();  
		for(ll i = head[u]; i!=-1; i = edge[i].nex)  
		{  
			ll v = edge[i].to;  
			if(sign[v]==-1 && edge[i].cap)  
			{  
				sign[v] = sign[u] + 1, q.push(v);  
				if(sign[to] != -1)return true;  
			}  
		}  
	}  
	return false;  
}  
ll Stack[N], top, cur[N];  
ll Dinic(ll from, ll to){
	ll ans = 0;  
	while( BFS(from, to) )  
	{  
		memcpy(cur, head, sizeof(head));  
		ll u = from;      top = 0;  
		while(1)  
		{  
			if(u == to)  
			{  
				ll flow = inf, loc;//loc 表示 Stack 中 cap 最小的边  
				for(ll i = 0; i < top; i++)  
					if(flow > edge[ Stack[i] ].cap)  
					{  
						flow = edge[Stack[i]].cap;  
						loc = i;  
					}  

					for(ll i = 0; i < top; i++)  
					{  
						edge[ Stack[i] ].cap -= flow;  
						edge[Stack[i]^1].cap += flow;  
					}  
					ans += flow;  
					top = loc;  
					u = edge[Stack[top]^1].to;  
			}  
			for(ll i = cur[u]; i!=-1; cur[u] = i = edge[i].nex)//cur[u] 表示u所在能增广的边的下标  
				if(edge[i].cap && (sign[u] + 1 == sign[ edge[i].to ]))break;  
			if(cur[u] != -1)  
			{  
				Stack[top++] = cur[u];  
				u = edge[ cur[u] ].to;  
			}  
			else  
			{  
				if( top == 0 )break;  
				sign[u] = -1;  
				u = edge[ Stack[--top]^1 ].to;  
			}  
		}  
	}  
	return ans;  
}
void init(){memset(head,-1,sizeof head);edgenum = 0;}

ll dis[N];
ll n, m;
ll mp[401][401];
bool inq[N];
void spfa(int from, int to){
	for(int i = 1; i <= n; i++)dis[i] = inf;
	memset(inq, 0, sizeof inq);
	dis[from] = 0;
	queue<int>q; q.push(from);
	inq[to] = 1;
	while(!q.empty()){
		int u = q.front(); q.pop(); inq[u] = 0;
		for(int i = 1; i <= n; i++)
			if(dis[i]>dis[u]+mp[u][i])
			{
				dis[i] = dis[u]+mp[u][i];
				if(!inq[i])inq[i] = 1, q.push(i);
			}

	}
}
void dfs(ll u, ll fa){
	if(u == n){printf("%d\n",u);return ;}
	else printf("%d ",u);
	for(ll i = head[u]; ~i; i = edge[i].nex){
		if(edge[i^1].cap != 1 || (i&1))continue;
		ll v = edge[i].to;
		if(v == fa)continue;
		edge[i^1].cap = 0;
		dfs(v, u);
		return;
	}
}
int main(){
	ll u, v, cost;
	while(~scanf("%d %d",&n,&m)){
		init();
		for(int i = 1; i <= n; i++)for(int j = 1; j <= n; j++)mp[i][j] = inf;
		while(m--)
		{
			scanf("%d %d %d",&u,&v,&cost);
			mp[u][v] = mp[v][u] = min(mp[u][v], cost);
		}
		spfa(1,n);
		if(dis[n] == inf){puts("No solution");continue;}
		for(ll i = 1; i <= n; i++)
			for(ll j = 1; j <= n; j++)if(mp[i][j]!=inf && dis[j] == mp[i][j] + dis[i])
				add(i,j,1);
		add(n,n+1,2);
		ll DIS = Dinic(1,n+1);
		if(DIS != 2){puts("No solution");continue;}
		else
		{
			dfs(1,1);
			dfs(1,1);
		}
	}
	return 0;
}
/*
4 6 
1 2 1 
1 3 1 
2 3 1 
3 4 2 
2 4 1 
1 4 1 

3 5 
1 3 1 
1 3 3  
1 3 3 
1 2 1 
2 3 1 

*/


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值