链接:https://ac.nowcoder.com/acm/contest/548/C
来源:牛客网
本题输入量较大,请注意使用效率较高的读入方式
输入的第一行包含四个整数 n, m, k, t,含义见上所述。
接下来一行,包含 k 个整数,表示立华奏已经学习过的知识点。如果 k=0,则此处为一空行。
接下来 m 行,每行 3 个整数 Xi,Yi,HiXi,Yi,Hi,描述一种联系。
如果立华奏能够学习完所有的知识点,输出一行 Yes。否则输出 No
0?k?n?106,m?5×106,t?1018,Ti,Hi?103
思路:
这就是一个最小生成树,只不过这题卡常数,因此从这题也学到了不少东西;
C | 运行超时 | 2001 | 0 | 1293 | C++ |
#include "bits/stdc++.h" using namespace std; typedef pair<int, int> PII; typedef long long LL; const int MAXN = 1e6 + 5; priority_queue<PII, vector<PII>, greater<PII> > que; vector<PII> vp[MAXN]; vector<int> vi; bool ok[MAXN]; void add(int k) { for (int i = 0; i < vp[k].size(); i++) { que.push(vp[k][i]); } vp[k].clear(); } int main() { int n, m, k, s; int a, b, c, cnt = 0; LL t, sum = 0; scanf("%d%d%d%lld", &n, &m, &k, &t); for (int i = 1; i <= n; i++) { scanf("%d", &s); que.push({s, i}); } for (int i = 1; i <= k; i++) { scanf("%d", &s); vi.push_back(s); } for (int i = 1; i <= m; i++) { scanf("%d%d%d", &a, &b, &c); vp[a].push_back({c, b}); vp[b].push_back({c, a}); } for (int i = 0; i < vi.size(); i++) { int j = vi[i]; if (ok[j] == false) { ok[j] = true; cnt++; add(j); } } while (cnt != n) { PII j = que.top(); que.pop(); if (ok[j.second] == false) { ok[j.second] = true; cnt++; sum += j.first; add(j.second); } } if (sum <= t) { puts("Yes"); } else { puts("No"); } return 0; }
类似Prim算法每次取最短边,但是比赛的时候我连这是最小生成树都没看出来,所以用了一个优先队列来维护目前能连起来的边,如果边的两个端点都未联通,则这条边在vector里不会加入优先队列,vector和优先队列转移耗费大量时间。而且各种耗时
C | 答案正确 | 1111 | 69944 | 1813 | C++ |
#include "bits/stdc++.h" using namespace std; typedef long long LL; typedef pair<int, int> PII; const int MAXN = 1e6 + 5; const int MAXW = 3000; vector<PII> vec[MAXW + 5]; int pre[MAXN], rak[MAXN]; int n, m, k; int u, v, w; LL t; inline LL read() { char c = getchar(); LL num = 0; while (!isdigit(c)) { c = getchar(); } while (isdigit(c)) { num = num * 10 + (c ^ ‘0‘); c = getchar(); } return num; } int find(int n) { if (pre[n] == -1) { return n; } return pre[n] = find(pre[n]); } bool check(int k) { for (w = 0; true; w++) { for (int i = 0; i < vec[w].size(); i++) { PII p = vec[w][i]; u = find(p.first); v = find(p.second); if (u != v) { t -= w; if (t < 0) { return false; } k++; if (t >= (n - k) * 1LL * MAXW) { return true; } if (rak[u] > rak[v]) { pre[v] = u; } else { pre[u] = v; if (rak[u] == rak[v]) { rak[v]++; } } } } } } int main() { n = read(), m = read(), k = read(), t = read();; memset(pre, -1, sizeof(pre)); memset(rak, 0, sizeof(rak)); for (int i = 1; i <= n; i++) { w = read(); vec[w].push_back({0, i}); } for (int i = 1; i <= k; i++) { u = read(); pre[u] = 0; } for (int i = 1; i <= m; i++) { u = read(); v = read(); w = read(); vec[w].push_back({u, v}); } if (check(k)) { puts("Yes"); } else { puts("No"); } return 0; }
在此之前就看到过快读(read),只是之前没有遇到像这题一样卡输入的。这题除非其他地方优化到极致,否则不用快读过不去。还有就是学到了并查集的启发式合并,之前写的并查集都只用了路径压缩来优化,才发现还可以用启发式合并来优化,而且就这题看来,用了启发式合并之后快了不少。还有就是学到了inline,之前没怎么用过这个关键字,网上查了一下说是可以提高代码效率(类似宏定义),不过关于这个inline,我试过去掉inline提交反而快了,不知道为什么。
链接:https://ac.nowcoder.com/acm/contest/548/C
来源:牛客网
本题输入量较大,请注意使用效率较高的读入方式
输入的第一行包含四个整数 n, m, k, t,含义见上所述。
接下来一行,包含 k 个整数,表示立华奏已经学习过的知识点。如果 k=0,则此处为一空行。
接下来 m 行,每行 3 个整数 Xi,Yi,HiXi,Yi,Hi,描述一种联系。
如果立华奏能够学习完所有的知识点,输出一行 Yes。否则输出 No
0?k?n?106,m?5×106,t?1018,Ti,Hi?103
nowcoder-548C-Tachibana Kanade Loves Review
原文:https://www.cnblogs.com/Angel-Demon/p/10678842.html