烂完了,读假又什么都不会呜呜呜。
要被自己菜哭了。
A - 133233#
很简单的签到。随便统计一下即可。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
#include <bits/stdc++.h>
using namespace std;
#define rep(i, a, b) for (auto i = (a); i <= decltype(i)(b); ++i)
inline void solve() {
int n; cin >> n;
vector<int> v(10, 0);
while (n) {
int x = n % 10; n /= 10;
if (x == 1) ++ v[1];
if (x == 2) ++ v[2];
if (x == 3) ++ v[3];
}
if (v[1] == 1 && v[2] == 2 && v[3] == 3) cout << "Yes" << char(10);
else cout << "No" << char(10);
}
signed main() {
cin.tie(nullptr)->sync_with_stdio(false);
int T; T = 1;
while (T--) solve();
return 0;
}
|
B - Hurdle Parsing#
签到 +1,扫一遍统计即可。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
#include <bits/stdc++.h>
using namespace std;
#define rep(i, a, b) for (auto i = (a); i <= decltype(i)(b); ++i)
inline void solve() {
string str; getline(cin, str);
vector<int> ans;
for (auto it : str) {
if (it == '|') ans.push_back(0);
if (it == '-') ++ans.back();
}
rep (i, 0, ans.size() - 2)
cout << ans[i] << " \n"[i == int(ans.size()) - 2];
}
signed main() {
cin.tie(nullptr)->sync_with_stdio(false);
int T; T = 1;
while (T--) solve();
return 0;
}
|
C - Move Segment#
有点讨厌的模拟。主要是 $1$ 子段的截取需要一点判断。
将所有 $1$ 子段的区间截取出来,然后调整其中的 $K$ 到 $K - 1$ 的后面即可。
时间复杂度 $\Theta(n)$
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
|
#include <bits/stdc++.h>
using namespace std;
#define rep(i, a, b) for (auto i = (a); i <= decltype(i)(b); ++i)
using pii = pair<int, int>;
inline void solve() {
int n, k; cin >> n >> k;
string str; cin >> str;
vector<pii> blk;
[&]() {
pii tmp; bool flg = false;
rep (i, 0, n - 1) {
if (i == 0) {
if (str[0] == '1') { tmp.first = 1; flg = 1; }
continue;
}
if (str[i] == '0') {
if (str[i - 1] == '1') {
tmp.second = i; flg = 0;
blk.push_back(tmp); tmp = {0, 0};
}
}
if (str[i] == '1') {
if (str[i - 1] == '0') { tmp.first = i + 1; flg = 1; }
}
}
if (flg == 1 && str.back() == '1') {
tmp.second = n; blk.push_back(tmp);
}
}();
int len = blk[k - 1].second - blk[k - 1].first;
blk[k - 1].first = blk[k - 2].second + 1;
blk[k - 1].second = blk[k - 1].first + len;
fill(str.begin(), str.end(), '0');
for (auto &it : blk)
fill(str.begin() + it.first - 1, str.begin() + it.second, '1');
cout << str << char(10);
}
signed main() {
cin.tie(nullptr)->sync_with_stdio(false);
int T; T = 1;
while (T--) solve();
return 0;
}
|
D - Strange Mirroring#
注意到 $K_i$ 的范围很大,故而肯定考虑类似 $2$ 的次幂相关的化归。
不妨先观察最终串的组成,写出前几项
$$
STTSTSST\ldots
$$
不难发现,在 $1 \sim 2^n$ 的范围中,$1 \sim 2^{n - 1}$ 和 $2^{n - 1} + 1 \sim 2^n$ 应该是互反的。
所以可以考虑如下算法:
对于当前的位置 $p$ 所在串,通过不断减去 $2^{\lfloor \log_2{p} \rfloor}$,记录减去后的反转情况,可以最终规约到串 $S$ 以及是否需要翻转。
由于该题时限猜测一定不会太严,并且不高兴操心 $long\space long$ 溢出问题,故代码中存在 define int long long 之流的恶习,请不要随意模仿。
代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
|
#include <bits/stdc++.h>
using namespace std;
#define int long long
#define rep(i, a, b) for (auto i = (a); i <= decltype(i)(b); ++i)
inline void solve() {
string s; cin >> s;
int q; cin >> q;
while (q--) {
int k; cin >> k;
int val = ((k-- + s.size() - 1) / s.size());
bool flg = false;
while (val > 1) {
int tmp = (1ll << (long long)(__lg(val - 1)));
val -= tmp;
k -= tmp * s.size();
flg ^= 1;
}
if (!flg) cout << s[k] << ' ';
else {
if (s[k] >= 97) cout << char(s[k] - 97 + 65) << ' ';
else cout << char(s[k] - 65 + 97) << ' ';
}
}
}
signed main() {
cin.tie(nullptr)->sync_with_stdio(false);
int T; T = 1;
while (T--) solve();
return 0;
}
|
额啊
我是冰茶姬实现低手 QwQ
赛时想到冰茶姬了,但是一度写错加读错题意 (size 函数返回 int 的笨比),所以小寄一波。
实际上这题个人觉得比较自然的想法是,使用类似珂朵莉树存放区间的方法,进行类似的合并和删除,实际上速度还挺快的(?
在这里放出两种不同的写法,感觉是加强了类似区间处理的一些想法。
Notice: 本段代码中 dsu 的 merge 函数非标准实现。珂朵莉树的节点定义中的 $operator<$ 重载需要加上两个 $const$,不然容易报神必错误。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
|
#include <bits/stdc++.h>
using namespace std;
#define rep(i, a, b) for (auto i = (a); i <= decltype(i)(b); ++i)
struct Node {
int l, r;
mutable int v;
Node(int l_, int r_ = -1, int v_ = 0) : l(l_), r(r_), v(v_) {};
bool operator < (const Node &o) const { return l < o.l; }
};
inline void solve() {
int n, q; cin >> n >> q;
set<Node> s;
rep (i, 1, n) { s.insert(Node(i, i, i)); }
vector<int> cnt(n + 10, 1);
while (q--) {
int opt; cin >> opt;
if (opt == 1) {
int x, c; cin >> x >> c;
auto it = s.lower_bound(Node(x, x, 0));
if (it->l != x) --it;
auto &[l, r, v] = *it;
cnt[v] -= r - l + 1; v = c; cnt[c] += r - l + 1;
bool flg1 = false, flg2 = false;
if (it != s.begin() && prev(it)->v == c) flg1 = true;
if (next(it) != s.end() && next(it)->v == c) flg2 = true;
if (flg1 || flg2) {
int nl = l, nr = r, nv = c;
if (flg1) { nl = prev(it)->l; s.erase(prev(it)); }
if (flg2) { nr = next(it)->r; s.erase(next(it)); }
s.erase(it); s.insert(Node(nl, nr, nv));
}
}
if (opt == 2) {
int c; cin >> c;
cout << cnt[c] << char(10);
}
}
}
signed main() {
cin.tie(nullptr)->sync_with_stdio(false);
int T; T = 1;
while (T--) solve();
return 0;
}
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
|
#include <bits/stdc++.h>
using namespace std;
#define rep(i, a, b) for (auto i = (a); i <= decltype(i)(b); ++i)
inline void solve() {
int n, q; cin >> n >> q;
vector<int> fa(n + 10), siz(n + 10, 1);
vector<int> cnt(n + 10, 1);
vector<int> l(n + 10), r(n + 10), c(n + 10);
iota(fa.begin(), fa.end(), 0);
rep (i, 1, n) l[i] = r[i] = c[i] = i;
auto find = [&](this auto&& find, int x) -> int {
return fa[x] == x ? x : fa[x] = find(fa[x]);
};
auto merge = [&](int x, int y) {
if (siz[x] < siz[y]) swap(x, y);
siz[x] += siz[y];
l[x] = min(l[x], l[y]);
r[x] = max(r[x], r[y]);
fa[y] = x;
};
while (q--) {
int opt; cin >> opt;
if (opt == 1) {
int x, c_; cin >> x >> c_;
int f = find(x);
cnt[c[f]] -= r[f] - l[f] + 1;
c[f] = c_;
cnt[c_] += r[f] - l[f] + 1;
if (l[f] != 1) {
int f_ = find(l[f] - 1);
if (c[f_] == c_) merge(f, f_);
}
if (r[f] != n) {
int f_ = find(r[f] + 1);
if (c[f_] == c_) merge(f, f_);
}
}
if (opt == 2) {
int c_; cin >> c_;
cout << cnt[c_] << char(10);
}
}
}
signed main() {
cin.tie(nullptr)->sync_with_stdio(false);
int T; T = 1;
while (T--) solve();
return 0;
}
|
提交结束后两者的速度差距差不多在 $4$ 倍左右,尚且合理。
F、G: To be continued.