Educational Codeforces Round 151 (Rated for Div. 2) A-D
A
代码
#include <bits/stdc++.h> using namespace std; using ll = long long; bool solve() { int n, k, x; cin >> n >> k >> x; if (x != 1) { cout << "YES" << '\n'; cout << n << '\n'; for (int i = 1;i <= n;i++) cout << 1 << " \n"[i == n]; } else { if ((n & 1) && n >= 3) { if (k <= 2) cout << "NO" << '\n'; else { cout << "YES" << '\n'; cout << (n - 3) / 2 + 1 << '\n'; cout << 3 << ' '; for (int i = 1;i <= n - 3;i += 2) cout << 2 << " \n"[i == n - 3]; } } else if (!(n & 1)) { if (k == 1) cout << "NO" << '\n'; else { cout << "YES" << '\n'; cout << n / 2 << '\n'; for (int i = 1;i <= n;i += 2) cout << 2 << " \n"[i == n - 1]; } } else cout << "NO" << '\n'; } return true; } int main() { std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0); int t = 1; cin >> t; while (t--) { if (!solve()) cout << -1 << '\n'; } return 0; }
B
代码
#include <bits/stdc++.h> using namespace std; using ll = long long; bool solve() { int xa, ya, xb, yb, xc, yc; cin >> xa >> ya >> xb >> yb >> xc >> yc; int ans = 1; if (1LL * (xa - xb) * (xa - xc) > 0) ans += min(abs(xa - xb), abs(xa - xc)); if (1LL * (ya - yb) * (ya - yc) > 0) ans += min(abs(ya - yb), abs(ya - yc)); cout << ans << '\n'; return true; } int main() { std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0); int t = 1; cin >> t; while (t--) { if (!solve()) cout << -1 << '\n'; } return 0; }
C
题意
给一个只包含数位的原串 ,以及长为 只包含数位的两个串 。
判断能否构造长为 的串,满足第 个数位在 内,且不是 的子序列。
题解
知识点:子序列自动机,贪心。
子序列自动机实际上就是 ,表示位置 往后(不包括 )最近一个字符 的位置,可以逆推的形式构造出来。
通过这个自动机我们构造串的每个数位,显然如果当前位置能找到一个接下来不存在的数位,那么我们必定选它,否则我们选择出现位置最晚的那个数位。这样贪心保证如果存在解,那我们一定能取到。
时间复杂度
空间复杂度
代码
#include <bits/stdc++.h> using namespace std; using ll = long long; int nxt[300007][10]; bool solve() { string s; cin >> s; int n = s.size(); s = "?" + s; int m; string l, r; cin >> m; cin >> l >> r; l = "?" + l; r = "?" + r; for (int i = 0;i <= 9;i++) nxt[n][i] = n + 1; for (int i = n - 1;i >= 0;i--) { for (int j = 0;j <= 9;j++) nxt[i][j] = nxt[i + 1][j]; nxt[i][s[i + 1] - '0'] = i + 1; } int pos = 0; for (int i = 1;i <= m;i++) { int mx = 0; for (int j = l[i] - '0';j <= r[i] - '0';j++) mx = max(mx, nxt[pos][j]); if (mx == n + 1) { cout << "YES" << '\n'; return true; } pos = mx; } cout << "NO" << '\n'; return true; } int main() { std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0); int t = 1; cin >> t; while (t--) { if (!solve()) cout << -1 << '\n'; } return 0; }
D
题意
给定一个长为 的整数数组 ,现在从左往右累和。
现在规定一个数 ,使得累和过程中若和一旦大于等于 ,那么之后不会再小于 。
找到一个 ,使得结果最大。
题解
知识点:前缀和,贪心。
的作用是使得让和小于 的位置,经过它们之后的结果等于 。所以规定一个 之后,所有会让经过后的和小于 的位置是可以无视的。
实际上,我们可以把第一个位置到最后一个位置当作连续的一段,一起无视。原因是,假设有两个使得结果会小于 的相邻位置 ,那么一定有经过 后结果是 ,之后再经过 之后结果也是 ,相当于从 的位置全部被无视了。
因此,对于一个 ,最终结果等于总和减去一段被无视的连续段的和,理论上答案最大不会超过总和减去最小和子段。问题变为,找到一个 ,其无视的一段连续段的和最小。
我们发现,实际上总是会有一个 ,能满足被他无视的一段就是最小和子段,使得结果是理论最大值。只需要找到整个数组最小和的那个连续段 ,随后 即为 的和。
结论的证明:
- 我们规定 为 ,并不会使得 的和改变,变得大于 ,而 即为 的和所经历的最大值。因为, 一定是 中最大的前缀和,否则若 是最大的前缀和,那么显然 的和一定是负的,则 为最小和子段,这就矛盾了。
- 在这个 下, 一定是被无视的一段,即经过 后结果还是 。首先结果不会小于 ,若结果大于 ,那么 一定存在一段后缀是正的,否则我们从 开始从 累加,一旦和为负则累和归零继续,因为不存在一段后缀是正的,最后结果累和一定为 ,结果也不可能大于 。因此,我们把这段后缀去掉会得到更小和的子段,与 是最小和子段矛盾。
- 而 是不会被无视的,如果有一段被无视,那么其一定有一个前缀是负的,同样可以类似反证,因此和 合并会得到更小和的子段,矛盾了。实际上,一开始证明答案的上限也可以得到这个结论。
综上,我们只需要找到最小和子段 ,并让 为 的和,那么 一定是有且仅有的被无视的一段,且 的和不会改变,那么结果即为总和减去最小和子段,是理论最大值。
以上整个过程,有两种实现方法:
- 枚举每个右端点 在 ,则 中的最大前缀和作为 , 取 最小值位置时的值。
- 枚举每个 ,则在 中找到最小前缀和(最大后缀和)作为 , 取 最小值位置时的值。
时间复杂度
空间复杂度
代码
#include <bits/stdc++.h> using namespace std; using ll = long long; bool solve() { int n; cin >> n; ll l = 0, r = 0, mi = 0; ll k = 0; for (int i = 1;i <= n;i++) { int x; cin >> x; r += x; l = max(l, r); if (r - l < mi) { mi = r - l; k = l; } } cout << k << '\n'; return true; } int main() { std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0); int t = 1; cin >> t; while (t--) { if (!solve()) cout << -1 << '\n'; } return 0; }
本文来自博客园,作者:空白菌,转载请注明原文链接:https://www.cnblogs.com/BlankYang/p/17524222.html
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧