Codeforces-1473-E. Minimum Path (分层图最短路)

题目

传送门

  • 简而言之,给一个有权无向图,起点为 1,问从 1 到所有其他点 分别 的最小“距离”,其中这个“距离”指路径长度减去最大边再加上最小边。
  • 点和边都是 2e5 范围。

思路

  • 可以类似 dp 一样将点拓展分层,一个点 x 分成4个:x[mx][mn],其中 mx 代表从 1 到这是否删掉了一个边, mn 代表从1到这是否加上了一个边(先不管最大最小)。然后连的边还是正常地连,在后面跑最短路的时候判断一下就行。
  • 那么在这个新的图中,最后的x[1][1]其实就是题目所求的“距离”因为刚好减去最大,加上最小,这样权值是最小的。
  • 分析一下时间复杂度:迪杰斯特拉最短路算法,用堆优化,O(4*N lg(N)) 大概吧。

代码

#include <queue>
#include <vector>
#include <cstring>
#include <cstdio>
using namespace std;
#define ID(x,y,z) (((x)<<3)+((y)<<1)+z)
const int N=200015;
 
vector<pair<int,int>> graph[N];
long long d[N][2][2];
bool vis[N][2][2];
 
int n,m;
 
struct cmp {
    
    
	bool operator() (pair<long long,long long> x, pair<long long,long long> y) {
    
    
		return x.first > y.first;
	}
};
 
 
int main()
{
    
    
	scanf("%d%d",&n,&m);
	for (int i=1,u,v,w; i<=m; i++) {
    
    
		scanf("%d%d%d",&u,&v,&w);
		u--;
		v--;
		graph[u].emplace_back(make_pair(v,w));
		graph[v].emplace_back(make_pair(u,w));
	}
	
	for (int i=0; i<n; i++) {
    
    
		for (int j=0; j<2; j++) {
    
    
			for (int k=0; k<2; k++) {
    
    
				d[i][j][k]=10000000000000000ll;
				vis[i][j][k]=0;
			}
		}
	}
	d[0][0][0]=0;
		
	priority_queue<pair<long long,long long>,vector<pair<long long,long long>>,cmp> q;
	q.push({
    
    0,ID(0,0,0)});
	
	while (!q.empty()) {
    
    
		int u,mx,mn;
		u=(q.top().second>>3);
		mx=(q.top().second>>1)&1;
		mn=q.top().second&1;
		q.pop();
		
		if (vis[u][mx][mn])
			continue;
		vis[u][mx][mn]=1;
		
		for (pair<int,int> I: graph[u]) {
    
    
			int v=I.first,w=I.second;
			for (int i=0; i<=(mx^1); i++) {
    
    
				for (int j=0; j<=(mn^1); j++) {
    
    
					if (d[v][mx|i][mn|j]>d[u][mx][mn]+1ll*(1-i+j)*w) {
    
    
						d[v][mx|i][mn|j]=d[u][mx][mn]+1ll*(1-i+j)*w;
						q.push(make_pair(d[v][mx|i][mn|j],ID(v,(mx|i),(mn|j))));
					}
				}
			}
		}
	}
	
	for (int i=1; i<n; i++) {
    
    
		printf("%lld ",d[i][1][1]);
	}
	puts("");
	
	
	return 0;
}
*/

猜你喜欢

转载自blog.csdn.net/jackypigpig/article/details/112730352
今日推荐