比赛链接


1487A. Arena

n 个 Hero,分别有 Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图1 的初始等级。每次两个 Hero 战斗时:等级相同无影响,否则等级高的英雄等级+1。直到某个英雄等级到了 Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图2游戏结束。问有多少个英雄最后可能获胜。

Solution

只要所有英雄等级不是相同,那么必有 Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图3个 Hero 获胜(Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图4)

Code

  1. int main() {
  2. ios_base::sync_with_stdio(false), cin.tie(0);
  3. for (cin >> _; _--;) {
  4. int n;
  5. cin >> n;
  6. vector<ll> a(n);
  7. for (auto& x : a) cin >> x;
  8. sort(a.begin(), a.end());
  9. int cnt = n, i = 0;
  10. while (a[i] == a[i + 1] && i < n) i++;
  11. if (i == n) {
  12. cout << 0 << "\n";
  13. continue;
  14. }
  15. cout << n - count(a.begin(), a.end(), a[0]) << "\n";
  16. }
  17. return 0;
  18. }

1478B. Cat Cycle

有一个Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图5个点的圆环。A在 Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图6 号点,B在 Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图7 号点。A逆时针走,B顺时针走。如果A和B下一步是一个点,那么A走一步,B走两步。问经过 Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图8 步后B在几号点

Solution

看了很久,发现是奇偶数问题,Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图9 为偶数不会碰到一起,奇数个点时 B 多走 Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图10

Code

  1. int main() {
  2. ios_base::sync_with_stdio(false), cin.tie(0);
  3. for (cin >> _; _--;) {
  4. ll n, k;
  5. cin >> n >> k;
  6. k--; // 第一秒
  7. if (n & 1) k += (k / (n / 2)); // 模拟奇数个点,k的变化
  8. k++, k %= n;
  9. cout << (k == 0 ? n : k) << "\n";
  10. }
  11. return 0;
  12. }

1478C. Minimum Ties

Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图11 支队伍要两两对战,平局各加一分,获胜加 3 分,失败无影响。请安排这些对局的结果使得最终所有人的分相等而且平局的数量尽可能少

Solution

显然总共有 Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图12%2F2#card=math&code=n%2A%28n-1%29%2F2) 局,如果全是非平局,那么一人 Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图13%2F2#card=math&code=%28n-1%29%2F2)局,那么只有在 Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图14 是奇数的情况下成立。不然必定有平局产生。

然后就按照每个人被分到多少局进行分配就好了。

  1. int main() {
  2. ios_base::sync_with_stdio(false), cin.tie(0);
  3. int n, i, j, k;
  4. for (cin >> n; cin >> n;)
  5. for (i = 0; i < n; i++)
  6. for (j = i + 1; k = (j - i) * 2, j < n; j++)
  7. cout << (k == n ? 0 : k < n ? 1 : -1) << ' ';
  8. return 0;
  9. }

1478D. Pythagorean Triples

寻找三元组(a,b,c)满足 Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图15 and Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图16

Solution

两个算式联立:Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图17 and Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图18

最后算出个:Educational Codeforces Round 104 (Rated for Div. 2) A-E 个人题解 - 图19((sqrt(2%20*%20n%20-%201)%20-%201)%20%2F%202)#card=math&code=%28ll%29%28%28sqrt%282%20%2A%20n%20-%201%29%20-%201%29%20%2F%202%29)

  1. int main() {
  2. ios_base::sync_with_stdio(false), cin.tie(0);
  3. for (cin >> _; _--;) {
  4. ll n;
  5. cin >> n;
  6. cout << (ll)((sqrt(2 * n - 1) - 1) / 2) << "\n";
  7. }
  8. return 0;
  9. }

1478E. Cheap Dinner

给定四个种类的菜各自的价格,给定一些关系:第一种菜中的某一种不能同时和第二种菜中的某一种一起被选中,然后2-nd和3-th,3-rd和4-th。

问四种菜中各择一的最小代价。

Solution

如果单考虑前两种,那么对于第二种菜中的所有菜,可以通过不同时被选中关系得到自己的最小代价(或者不能),把这个代价移到第二种菜上,2-nd和3-rd的菜之间的转移等价于1-st和2-nd之间的转移。一种类似DP转移的思想。代码实现方面就用multiset乱搞,每次考虑两种关系的时候,先把当前点所不连通的点删掉,最后再加回来,可以这么做的原因是我只要考虑值,不需要考虑具体的关系。

Code

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. const int maxn = 2e5 + 10;
  4. int arr[5][maxn], dp[5][maxn];
  5. int nums[5];
  6. vector<int> vv[5][maxn];
  7. void solve() {
  8. for (int i = 1; i <= 4; ++i) scanf("%d", &nums[i]);
  9. for (int i = 1; i <= 4; ++i) {
  10. for (int j = 1; j <= nums[i]; ++j) {
  11. scanf("%d", &arr[i][j]);
  12. }
  13. }
  14. for (int i = 2; i <= 4; ++i) {
  15. int m;
  16. scanf("%d", &m);
  17. for (int j = 1, x, y; j <= m; ++j) {
  18. scanf("%d %d", &x, &y);
  19. vv[i][y].push_back(x);
  20. }
  21. }
  22. for (int i = 1; i <= nums[1]; ++i) {
  23. dp[1][i] = arr[1][i];
  24. }
  25. for (int i = 2; i <= 4; ++i) {
  26. multiset<int> ss;
  27. for (int l = 1; l <= nums[i - 1]; ++l) {
  28. if (dp[i - 1][l] != -1) ss.insert(dp[i - 1][l]);
  29. }
  30. for (int j = 1; j <= nums[i]; ++j) {
  31. for (int k = 0; k < vv[i][j].size(); ++k) {
  32. int v = vv[i][j][k];
  33. if (dp[i - 1][v] != -1) ss.erase(ss.find(dp[i - 1][v]));
  34. }
  35. if (ss.size() == 0)
  36. dp[i][j] = -1;
  37. else
  38. dp[i][j] = *ss.begin() + arr[i][j];
  39. for (int k = 0; k < vv[i][j].size(); ++k) {
  40. int v = vv[i][j][k];
  41. if (dp[i - 1][v] != -1) ss.insert(dp[i - 1][v]);
  42. }
  43. }
  44. }
  45. int ans = -1;
  46. for (int i = 1; i <= nums[4]; ++i) {
  47. if (dp[4][i] != -1) {
  48. if (ans == -1 || ans > dp[4][i]) ans = dp[4][i];
  49. }
  50. }
  51. printf("%d\n", ans);
  52. }
  53. int main() {
  54. int t = 1;
  55. // scanf("%d", &t);
  56. while (t--) solve();
  57. return 0;
  58. }