[ABC270F] Transportation
[ABC270F] Transportation
题意
有
思路
显然是最小生成树,但是由于可以见两种东西,所以比较难处理,所以可以将这个点权变成一条边,这样就好处理了,所以可以额外开两个点
代码
#include <algorithm> #include <iostream> using namespace std; const int MaxN = 2e5 + 10; struct S { int u, v, w; bool operator<(const S &j) const { return w < j.w; } } a[MaxN + 2 * MaxN]; int n, m, tot; long long fa[MaxN], ans = 1e18; bool vis[MaxN]; int FindFather(int x) { return fa[x] < 0 ? x : fa[x] = FindFather(fa[x]); } void insert(int x, int y) { x = FindFather(x), y = FindFather(y); if (fa[x] < fa[y]) { swap(x, y); } fa[y] += fa[x], fa[x] = y; } int main() { cin >> n >> m; for (int i = 1, u; i <= n; i++) { cin >> u; a[++tot] = {i, n + 1, u}; } for (int i = 1, u; i <= n; i++) { cin >> u; a[++tot] = {i, n + 2, u}; } for (int i = 1, u, v, w; i <= m; i++) { cin >> u >> v >> w; a[++tot] = {u, v, w}; } sort(a + 1, a + tot + 1); for (int u = 0; u < 4; u++) { fill(fa + 1, fa + n + 4, -1); fill(vis + 1, vis + n + 4, 0); long long sum = 0; for (int i = 1; i <= tot; i++) { if ((a[i].v == n + 1 && ((u >> 1) & 1) != 1) || (a[i].v == n + 2 && (u & 1) != 1)) { continue; } if (FindFather(a[i].u) != FindFather(a[i].v)) { sum += a[i].w, vis[a[i].u] = vis[a[i].v] = 1; insert(a[i].u, a[i].v); } } int cnt = 0; for (int i = 1; i <= n + 4; i++) { cnt += vis[i]; } if (u == 0) { if (cnt != n) { sum = 1e18; } } else if (u == 1 || u == 2) { if (cnt != n + 1) { sum = 1e18; } } else if (u == 3) { if (cnt != n + 2) { sum = 1e18; } } ans = min(ans, sum); } cout << ans << endl; return 0; }
本文作者:yabnto
本文链接:https://www.cnblogs.com/ybtarr/p/17410156.html
版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步