Codeforces Round 977 (Div. 2)E1 Digital Village (Easy Version)(Floyd,贪心)

题目链接

Codeforces Round 977 (Div. 2)E1 Digital Village (Easy Version)

思路

首先,我们注意到 n n n的最大值只有 400 400 400。

因此,我们可以先用 F l o y d Floyd Floyd算法预处理出任意两座城市之间的最大延迟时间。

之后,我们通过在线操作,每次贪心地选出最优的一个城市,并不断更新答案。

即,我们先选出 k = 1 k=1 k=1时的最优解,之后从剩下的点里面挑出一个能够使 k = 2 k=2 k=2时最优的点,以此类推。

代码

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;
#define int long long
const int N = 4e2 + 5;
const int mod = 998244353;
const int inf = 0x3f3f3f3f3f3f3f3f;
int n, m, p;
void solve()
{
	cin >> n >> m >> p;
	vector<int>city(p);
	for (int &val : city)
	{
		cin >> val;
	}
	vector<vector<int>>dist(n + 1, vector<int>(n + 1, inf));
	for (int i = 1, u, v, w; i <= m; i++)
	{
		cin >> u >> v >> w;
		dist[u][v] = min(dist[u][v], w);
		dist[v][u] = min(dist[v][u], w);
	}

	for (int i = 1; i <= n; i++)
		dist[i][i] = 0;

	for (int k = 1; k <= n; k++)
	{
		for (int i = 1; i <= n; i++)
		{
			for (int j = 1; j <= n; j++)
			{
				dist[i][j] = min(dist[i][j], max(dist[i][k], dist[k][j]));
			}
		}
	}


	vector<int>ans(n + 1);

	int sum = inf, idx = -1;
	for (auto u : city)
	{
		int res = 0;
		for (auto v : city)
		{
			res += dist[u][v];
		}
		if (sum >= res)
		{
			idx = u;
			sum = res;
			for (auto v : city)
			{
				ans[v] = dist[u][v];
			}
		}
	}

	set<int>st;
	st.insert(idx);

	cout << sum << " ";

	for (int i = 2; i <= n; i++)
	{
		if (sum == 0)
		{
			cout << sum << " ";
			continue;
		}
		int maxx = 0;
		idx = -1;
		for (auto u : city)
		{
			if (st.count(u)) continue;
			int res = 0;
			for (auto v : city)
			{
				if (st.count(v)) continue;
				if (ans[v] > dist[u][v])
				{
					res += ans[v] - dist[u][v];
				}
			}
			if (maxx <= res)
			{
				maxx = res;
				idx = u;
			}
		};
		st.insert(idx);
		sum -= maxx;
		for (auto v : city)
		{
			if (st.count(v)) continue;
			ans[v] = min(ans[v], dist[idx][v]);
		}
		cout << sum << " ";
	}
	cout << endl;
}
signed main()
{
	ios::sync_with_stdio(false);
	cin.tie(0), cout.tie(0);
	int test = 1;
	cin >> test;
	for (int i = 1; i <= test; i++)
	{
		solve();
	}
	return 0;
}
相关推荐
Tisfy几秒前
LeetCode 3315.构造最小位运算数组 II:位运算
算法·leetcode·题解·位运算
楼田莉子5 分钟前
Linux学习之库的原理与制作
linux·运维·服务器·c++·学习
YuTaoShao14 分钟前
【LeetCode 每日一题】1292. 元素和小于等于阈值的正方形的最大边长
算法·leetcode·职场和发展
Remember_99314 分钟前
【数据结构】深入理解Map和Set:从搜索树到哈希表的完整解析
java·开发语言·数据结构·算法·leetcode·哈希算法·散列表
浅念-16 分钟前
C++第一课
开发语言·c++·经验分享·笔记·学习·算法
charlie11451419119 分钟前
现代嵌入式C++教程:对象池(Object Pool)模式
开发语言·c++·学习·算法·嵌入式·现代c++·工程实践
HABuo21 分钟前
【linux进程控制(三)】进程程序替换&自己实现一个bash解释器
linux·服务器·c语言·c++·ubuntu·centos·bash
TTGGGFF25 分钟前
控制系统建模仿真(二):掌握控制系统设计的 MAD 流程与 MATLAB 基础运算
开发语言·数据结构·matlab
燃于AC之乐31 分钟前
我的算法修炼之路--8——预处理、滑窗优化、前缀和哈希同余,线性dp,图+并查集与逆向图
算法·哈希算法·图论·滑动窗口·哈希表·线性dp
格林威42 分钟前
多相机重叠视场目标关联:解决ID跳变与重复计数的 8 个核心策略,附 OpenCV+Halcon 实战代码!
人工智能·数码相机·opencv·算法·计算机视觉·分类·工业相机