CF666B World Tour

World Tour の 传送门

\(4 \le n \le 3000\)

说明可以用 \(n^2\) 的做法,题目要求 \(4\) 个点的最短路最长,共 \(3\) 条路经,则枚举 \(2\) 个点。

如果枚举 \(a, c\),则要找 \(b, d\),但 \(b\)\(c\) 也要判断路径,比较麻烦,所以直接枚举 \(b, c\)

然后枚举 \(b, c\) 对应的最短路最长的点,因为枚举点共会 \(6\) 种重合情况,而 \(b, c\) 已经判掉了,所以只有 \(5\) 种情况,就直接枚举最短路最长的前 \(5\)\(a\)\(d\) 同理。

#include <iostream>
#include <algorithm>
#include <cstring>
#include <utility>
#include <vector>
#include <queue>
#define d first
#define id second
using namespace std;
constexpr int N = 3e3 + 5;
int n, m, vis[N];
vector<int> g[N];
int pre[N][N], dx[N][N];
pair<int, int> dis[N][N];
inline void dij(int s) {
	memset(dis[s], -1, sizeof(dis[s]));
	dis[s][s].d = 0;
	memset(dx[s], -1, sizeof(dx[s]));
	dx[s][s] = 0;
	for (int i = 1; i <= n; ++i)
		dis[s][i].id = i;
	priority_queue<pair<int, int>, vector<pair<int, int>>, greater<pair<int, int>>> q;
	q.push({0, s});
	while (!q.empty()) {
		int u = q.top().id;
		q.pop();
		vis[u] = 1;
		for (int l = 0; l < g[u].size(); ++l) {
			int i = g[u][l];
			if (dis[s][i].d > dis[s][u].d + 1 || dis[s][i].d == -1) {
				dis[s][i].d = dis[s][u].d + 1;
				dx[s][i] = dx[s][u] + 1;
				if (!vis[i])	q.push({dis[s][i].d, i});
			}
		}
	}
	sort(dis[s] + 1, dis[s] + n + 1, greater<pair<int, int>>());
}
int id;
inline bool cmp(int a, int b) {
	return dx[a][id] > dx[b][id];
}
inline void special() {
	for (int i = 1; i <= n; ++i) {
		for (int j = 1; j <= n; ++j)
			if (dx[j][i] != -1)	pre[i][j] = j;
		id = i;
		sort(pre[i] + 1, pre[i] + n + 1, cmp);
	}
}
signed main() {
	cin >> n >> m;
	while (m--) {
		int u, v;
		cin >> u >> v;
		g[u].emplace_back(v);
	}
	for (int i = 1; i <= n; ++i) {
		memset(vis, 0, sizeof(vis));
		dij(i);
// 		cerr << "DIJ " << i << '\n';
// 		for (int j = 1; j <= n; ++j)
// 			cerr << "  " << dis[i][j].id << ' ' << dis[i][j].d << '\n';
	}
	special();
	int ans = 0, anss[5];
	for (int j = 1; j <= n; ++j)
		for (int k = 1; k <= n; ++k) {
			if (j == k || dx[j][k] == -1)	continue;
			for (int ix = 1; ix <= min(3, n); ++ix) {
				int i = pre[j][ix];
				if (i == j || k == i || i == 0)	continue;
				for (int lx = 1; lx <=  min(3, n); ++lx) {
					int l = dis[k][lx].id;
					if (i == l || k == l || j == l || l == 0)	continue;
					if (ans < dx[i][j] + dx[j][k] + dx[k][l]) {
						ans = dx[i][j] + dx[j][k] + dx[k][l];
//						printf("YEAH %d %d %d %d\n  %d %d %d\n", i, j, k, l, dx[i][j], dx[j][k], dx[k][l]);
						anss[1] = i, anss[2] = j, anss[3] = k, anss[4] = l;
					}
				}
			}
		}
	for (int i = 1; i <= 4; ++i)
		cout << anss[i] << ' ';
	return 0;
}
posted @ 2023-10-04 10:35  SilverLi  阅读(24)  评论(0编辑  收藏  举报