构造C的歪
思路
取 \(|a-b|+\max(a,b)\) 即可构造第三项。
代码
#include <bits/stdc++.h>using namespace std;using i64 = long long;int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int a, b;cin >> a >> b;int d = abs(a-b);cout << max(a,b) + d << "\n";return 0;
}
不要三句号的歪
思路
采用C语言 \(scanf\) 的模式串读取即可得到 \(a,b,c\),然后输出 \(c-b-1\) 即可。
代码
#include <bits/stdc++.h>using namespace std;using i64 = long long;int main() {ios::sync_with_stdio(false);cin.tie(nullptr);i64 a, b, c;scanf("%lld,%lld,...,%lld", &a, &b, &c);printf("%lld\n", c - b - 1);return 0;
}
仰望水面的歪
思路
实际上就是将坐标按照水面对称后得到新的点与原点建立向量即可。
代码
#include <bits/stdc++.h>using namespace std;using i64 = long long;int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n, h;cin >> n >> h;while (n --) {i64 x, y, z;cin >> x >> y >> z;z = h + abs(h - z);i64 nx = x, ny = y, nz = z;i64 g = gcd(nx, gcd(ny, nz));nx /= g,ny /= g,nz /= g;cout << nx << " " << ny << " " << nz << "\n";}return 0;
}
小心火烛的歪
思路
注意到 \(p\le 7\),那么我们可以直接二进制枚举每个计划选还是不选即可,对选出的计划处理出最终的方案是否与草地互补即可。
代码
#include <bits/stdc++.h>using namespace std;using i64 = long long;int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n, m, q;cin >> n >> m >> q;vector<string> cs(n);for (auto &i : cs) {cin >> i;for (auto &j : i) {j = ((j - '0') ^ 1) + '0';}}vector has(q, vector<string>(n));for (auto &i : has) {for (auto &j : i) {cin >> j;}}int ans = -1;vector<int> res;auto check = [&](int x)->void{vector<string> now(n, string(m, '0'));vector<int> ok;for (int i = 0; i < q; i ++) {if (x >> i & 1) {ok.emplace_back(i + 1);for (int j = 0; j < n; j ++) {for (int k = 0; k < m; k ++) {if (has[i][j][k] == '1') {now[j][k] = '1';}}}}}if (now == cs) {if (ans == -1 || ok.size() < ans) {ans = ok.size();swap(res, ok);}}};for (int i = 0; i < (1 << q); i ++) {check(i);}cout << ans << "\n";for (auto i : res) {cout << i << " ";}return 0;
}
喜欢切数组的红
思路
要分成和相等的三部分,那么总体的和也一定是三的倍数。
然后做个前缀和,对达到 \(\frac{sum}{3}\) 的再去遍历后续是否存在满足要求的划分即可。
赛时没想过这样暴力,后来看别人代码这样过了,我想这样可以做的原因大概是因为能满足前缀和等于 \(\frac{sum}{3}\) 的条件很少,所以第二个循环并不会跑太多次(但是假如塞一堆 \(0\) 去增加条件 \(1\) 的判定不知道会不会超时,不懂)。
代码
#include <bits/stdc++.h>using namespace std;using i64 = long long;int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n;cin >> n;vector<i64> pre(n + 1), has(n + 1);vector<int> a(n + 1);for (int i = 1; i <= n; i ++) {cin >> a[i];pre[i] += pre[i - 1] + a[i];has[i] += has[i - 1] + (a[i] > 0);}if (pre[n] % 3 != 0 || has[n] < 3) {cout << 0 << "\n";return 0;}i64 avg = pre[n] / 3, ans = 0;for (int i = 1; i <= n; i ++) {if (!has[i] || pre[i] != avg) {continue;}for (int j = i + 1; j <= n; j ++) {if (pre[j] - pre[i] != avg || has[j] - has[i] == 0) {continue;}if (pre[n] - pre[j] != avg || has[n] - has[j] == 0) {continue;}ans ++;}}cout << ans << "\n";return 0;
}
研究red子序列的红
思路
唐了,刚开始看了一眼还以为树状数组能写,然后写到后面越发觉得不对劲,没想到线段树去维护,还是得多写orz
求区间内(实际上就是\([1,n]\))的子序列 red
的数量,对于交换操作,实际上就是单点修改,所以可以用线段树维护 r
,e
,d
,re
,ed
,red
\(6\) 个子串的个数,往上传的时候 re
=r
\(\times\) e
+左儿子re
+右儿子re
,其他同理。
代码
#include <bits/stdc++.h>using namespace std;using i64 = long long;template<class Node>
struct SegmentTree {
#define lc u<<1
#define rc u<<1|1const int n, N;vector<Node> tr;SegmentTree(): n(0) {}SegmentTree(int n_): n(n_), N(n * 4 + 10) {tr.reserve(N);tr.resize(N);}SegmentTree(string &init) : SegmentTree(init.size() - 1) {function<void(int, int, int)> build = [&](int u, int l, int r) {tr[u].l = l, tr[u].r = r;if (l == r) {tr[u] = {l, r, {init[l] == 'r', init[l] == 'e', init[l] == 'd', 0, 0, 0}};return ;}i64 mid = (l + r) >> 1;build(lc, l, mid);build(rc, mid + 1, r);pushup(tr[u], tr[lc], tr[rc]);};build(1, 1, n);}void pushup(Node& U, Node& L, Node& R) { //上传U.l = L.l, U.r = R.r;for (int i = 0; i < 6; i ++) {U.res[i] = L.res[i] + R.res[i];}U.res[3] += L.res[0] * R.res[1];U.res[4] += L.res[1] * R.res[2];U.res[5] += L.res[0] * R.res[4] + L.res[3] * R.res[2];}void modify(int u, int x, char c, int k) {if (tr[u].l >= x && tr[u].r <= x) {if (c == 'r') tr[u].res[0] += k;else if (c == 'e') tr[u].res[1] += k;else if (c == 'd') tr[u].res[2] += k;return ;}int mid = (tr[u].l + tr[u].r) >> 1;if (x <= mid)modify(lc, x, c, k);if (x > mid)modify(rc, x, c, k);pushup(tr[u], tr[lc], tr[rc]);}i64 get() {return tr[1].res[5];}
};struct Node { //线段树定义int l, r;array<i64, 6> res;
};int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n, q;cin >> n >> q;string s, t;cin >> s >> t;s = " " + s, t = " " + t;string pa = "red";SegmentTree<Node> st(s), tt(t);while (q--) {int x;cin >> x;if (pa.find(s[x]) != -1) {st.modify(1, x, s[x], -1);}if (pa.find(t[x]) != -1) {tt.modify(1, x, t[x], -1);}swap(s[x], t[x]);if (pa.find(s[x]) != -1) {st.modify(1, x, s[x], 1);}if (pa.find(t[x]) != -1) {tt.modify(1, x, t[x], 1);}cout << st.get() - tt.get() << "\n";}return 0;
}