惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

SecWiki News
SecWiki News
I
InfoQ
The Cloudflare Blog
人人都是产品经理
人人都是产品经理
博客园 - Franky
T
Tailwind CSS Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
量子位
博客园_首页
罗磊的独立博客
V
V2EX
李成银的技术随笔
大猫的无限游戏
大猫的无限游戏
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
T
True Tiger Recordings
Vercel News
Vercel News
Cyberwarzone
Cyberwarzone
Cisco Talos Blog
Cisco Talos Blog
F
Fox-IT International blog
D
Darknet – Hacking Tools, Hacker News & Cyber Security
M
Microsoft Research Blog - Microsoft Research
Know Your Adversary
Know Your Adversary
爱范儿
爱范儿
The Register - Security
The Register - Security
G
Google Developers Blog
The Hacker News
The Hacker News
Malwarebytes
Malwarebytes
S
Securelist
博客园 - 三生石上(FineUI控件)
Jina AI
Jina AI
T
Threat Research - Cisco Blogs
T
The Exploit Database - CXSecurity.com
S
SegmentFault 最新的问题
博客园 - 叶小钗
F
Fortinet All Blogs
Apple Machine Learning Research
Apple Machine Learning Research
宝玉的分享
宝玉的分享
博客园 - 聂微东
T
Threatpost
博客园 - 【当耐特】
D
Docker
P
Privacy & Cybersecurity Law Blog
www.infosecurity-magazine.com
www.infosecurity-magazine.com
G
GRAHAM CLULEY
V
Visual Studio Blog
C
Cisco Blogs
IT之家
IT之家
S
Security Archives - TechRepublic
Latest news
Latest news
阮一峰的网络日志
阮一峰的网络日志

Shiroha白羽的博客

Codeforces Round 1089 (Div. 2) Codeforces Round 1079 (Div. 2) 2025 总结 The Burrows-Wheeler Transform 块排序压缩算法 Summer Pockets 动画完结 香港银行开卡记-1 【转载】Rust 中常见的有关生命周期的误解 Codeforces Round 972 (Div. 2) 《黑神话:悟空》游玩评测 Golang 踩坑 —— interface 为参数的时候传 nil 指针 用 magic 变量解决 UAF 问题 Codeforces Round 942 (Div. 2) Codeforces Round 941 (Div. 2) Codeforces Round 940 (Div. 2) and CodeCraft-23 日本旅游杂记-东京篇 Codeforces Round 939 (Div. 2) Codeforces Round 928 (Div. 4) Codeforces Round 927 (Div. 3) Codeforces Round 926 (Div. 2) Codeforces Round 925 (Div. 3) Codeforces Round 924 (Div. 2) Codeforces Round 923 (Div. 3) Codeforces Round 922 (Div. 2) Codeforces Round 921 (Div. 2) Educational Codeforces Round 161 (Rated for Div. 2) Codeforces Round 920 (Div. 3) Codeforces Round 919 (Div. 2) Hello 2024 Good Bye 2023 Codeforces Round 918 (Div. 4) 个人备份的常用 macOS 清理命令 Codeforces Round 917 (Div. 2) Pinely Round 3 (Div. 1 + Div. 2) Educational Codeforces Round 160 (Rated for Div. 2) Codeforces Round 915 (Div. 2) Codeforces Round 914 (Div. 2) Codeforces Round 913 (Div. 3) Educational Codeforces Round 159 (Rated for Div. 2) Codeforces Round 912 (Div. 2) Codeforces Round 911 (Div. 2) CodeTON Round 7 (Div. 1 + Div. 2, Rated, Prizes!) Educational Codeforces Round 158 (Rated for Div. 2) Codeforces Round 910 (Div. 2) Codeforces Round 909 (Div. 3) Codeforces Round 908 (Div. 2) Educational Codeforces Round 157 (Rated for Div. 2) C++自定义的字面量 Codeforces Round 907 (Div. 2) Codeforces Round 916 (Div. 3) 关于 LRU map 的一些灵感 2023 杭州站 ICPC 现场赛 反复横跳的 Clang-Tidy(cert-dcl21-cpp) Codeforces Round 906 (Div. 2) 一段奇怪的 CPP 代码 Codeforces Round 905 (Div. 3) Codeforces Round 904 (Div. 2) Codeforces Round 903 (Div. 3) Educational Codeforces Round 156 (Rated for Div. 2) Codeforces Round 902 (Div. 2, based on COMPFEST 15 - Final Round) Codeforces Round 901 (Div. 2) Codeforces Round 900 (Div. 3) Codeforces Round 899 (Div. 2) Educational Codeforces Round#155 (Div. 2) Codeforces Round 898 (Div. 4) CodeTON Round 6 (Div. 2) Codeforces Round 897 (Div. 2) Codeforces Round 896 (Div. 2) Codeforces Round 887 (Div. 2) Codeforces Round 895 (Div. 3) 左值-右值-将亡值 Educational Codeforces Round#154 (Div. 2) Pinely Round 2 (Div. 1 + Div. 2) Harbour.Space Scholarship Contest 2023-2024 (Div. 1 + Div. 2) Codeforces Round 894 (Div. 3) Codeforces Round 888 (Div. 3) Educational Codeforces Round#153 (Div. 2) OTPAUTH,两步验证中的通用协议 Codeforces Round 892 (Div. 2) Codeforces Round 891 (Div. 3) Codeforces Round 890 (Div. 2) Educational Codeforces Round#152 (Div. 2) Codeforces Round#889 (Div. 2) Java Script 的 null 和 undefined 随想 记一次 SQL LEFT JOIN 没有得到预期结果的错误 Codeforces Round#789(Div. 2) GCC/G++ 预编译头性能优化 使用 Junit5 和 Mockito 实现 SpringBoot 的单元测试最优美的解决方案 centOS 防火墙 docker-compse 的问题 Gitbook 安装出错 macOS 更新后导致 sdk 丢失问题 Java 生成验证码 Captcha C++ 模版可变参数列表传递给 C 的 va_list 可变参数列表 GitHub 下载的 zip 代码如何与原仓库再次建立连接 2021年浙江工商大学新生赛题解 Dockerfile 中下载 JDK8 Mac 使用带 python 的 vim Mac 截图唤起速度慢 C++ 语言实现动态变化的线程池 Strassen算法代码 Codeforces Round#744 (Div. 3)
Codeforces Round 893 (Div. 2)
2023-08-19 · via Shiroha白羽的博客

A. Buttons

大致题意

有 $1, 2, 3$ 三种按钮,其中 Anna 只能按 $1, 3$ 两种按钮,而 Katie 只能按 $2, 3$ 两种按钮。每个按钮只能按一次。

Anna 和 Katie 玩游戏,两人依次按按钮,Anna 先,直到谁没有按钮可以按,谁就输了,问谁会赢

思路

明显大家先抢着把 $3$ 按完就行了,因为 Anna 先开始按,所以为偶数则恰好对半分,为奇数则 Anna 多分到一个,然后计算谁按钮多就行了

AC code

1
2
3
4
5
6
7
8
9
10
11
void solve() {
int _;
cin >> _;
for (int ts = 0; ts < _; ++ts) {
int a, b, c;
cin >> a >> b >> c;
a += (c + 1) / 2;
b += c / 2;
cout << (a > b ? "First" : "Second") << endl;
}
}

B. The Walkway

大致题意

有一条路,路上有一些饼干店,饼干店的初始位置都确定,有一个人带着无限数量的饼干,从路的某一端匀速走到另外一端,每隔 $k$ 分钟没有吃饼干的情况下,他会吃掉背包里的一片饼干,如果刚刚遇到了饼干店的情况下,他也会吃掉一片饼干,在同一个时间点不会吃掉超过一片饼干,且在刚刚进入路的时候需要吃一片饼干。

你可以移除掉一个,最多一个饼干店,问最少只需要吃到多少饼干

思路

通过饼干店,可以分割成 $n$ 段,每一段吃掉的饼干数量等于 $\left \lceil \frac{pos_{n} - pos_{n-1}}{t} \right \rceil$,那么对于每一个饼干店 $i$,若其被移除掉,可以带来减少的饼干数量为 $\left \lceil \frac{pos_{i + 1} - pos_{i - 1}}{t} \right \rceil - \left \lceil \frac{pos_{i} - pos_{i - 1}}{t} \right \rceil - \left \lceil \frac{pos_{i + 1} - pos_{i}}{t} \right \rceil$

所以只需要枚举所有可能被干掉的饼干店,找到能减少的最大值就行了,注意一下,这里输出的最小饼干数量和能满足这个最小饼干数量的饼干店数量,而不是唯一那个位置

AC code

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
void solve() {
int _;
cin >> _;
for (int ts = 0; ts < _; ++ts) {
int n, m, d;
cin >> n >> m >> d;
vector<int> data(m);
for (int i = 0; i < m; ++i) cin >> data[i];

int tot = 0, ans = 0, cnt = 0;
int l = 1, r = data[1];
for (int i = 0; i < m; ++i) {
int tmp = (data[i] - l + d - 1) / d;
tot += tmp;
tmp += (r - data[i] + d - 1) / d;
int del = (r - l + d - 1) / d;
if (tmp - del > ans) {
cnt = 1;
ans = tmp - del;
} else if (tmp - del == ans) cnt++;
l = data[i];
r = i + 2 < m ? data[i + 2] : n + 1;
}
tot += (r - data[m - 1] + d - 1) / d;
cout << tot - ans << ' ' << cnt << endl;
}
}

C. Yet Another Permutation Problem

大致题意

给出一个 $n$ 的排列,使得所有相邻两个数的 $GCD$ 的值的种类足够多

思路

若相邻两个数字存在 $GCD$,那么必然这个 $GCD$ 小于等于其中的任意一个。而因为恰好是 $n$ 的排列,那么必然此 $GCD$ 的值本身一定存在与序列中。那么如果我直接取每个值以及其两倍的值放在一起,那么必然可以保证这个值可以存在。而且每个值的一半一定唯一,那么就可以得到唯一确认的绑定关系,然后排列即可

AC code

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
void solve() {
int _;
cin >> _;
for (int ts = 0; ts < _; ++ts) {
int n;
cin >> n;
vector<bool> vis(n + 1, false);
vector<int> res;
res.reserve(n);
for (int i = 1; i <= n; ++i) {
if (vis[i]) continue;
res.push_back(i);
vis[i] = true;
int tmp = i * 2;
while (tmp <= n) {
vis[tmp] = true;
res.push_back(tmp);
tmp *= 2;
}
}
for (int i = 0; i < n; ++i) cout << res[i] << " \n"[i == n - 1];
}
}

D. Trees and Segments

大致题意

有一个长度为 $n$ 的 $01$ 串,允许你翻转其中 $k$ 个,求对于每个 $a \in [1, n]$,求算 $a \times len_0 + len_1$ 的最大值。

$len_x$ 的指在这个字符串内,最长的一段连续的 $x$ 的长度

思路

这道题的难度跃升有点快,确实很难想清楚

首先考虑最暴力的情况,遍历每种 $a$,遍历所有可能的最长的 $0$ 的左右区间和 $1$ 的左右区间,然后计算是否满足并求解,那么总共需要 $n^6$

首先校验合法可以通过前缀和的方式预处理,这样就可以少一个 $n$

再考虑到不同的 $a$ 之和长度有关,而长度最多只有 $n$ 种可能(当 $len_0$ 为 $x$ 的时候,$max(len_1)$ 一定是唯一解),所以也不需要遍历所有 $a$,只需要计算出所有可能,然后再让 $a$ 和所有可能进行遍历即可,那么只需要一个单独的 $n^2$。

这样,我们只剩下了 $n^4$

为了达到目标,我们还需要拆分这两个 $n^2$,让找 $max(len_1)$ 变成近乎 $O(1)$ 的查找。那么很显然我们需要预处理,因为当前通过 $n^2$ 的方式确定 $len_0$ 的情况下,其实 $len_1$ 仅会出现在这个区间的左边或者右边,故预处理从 $1 \rightarrow n$ 的每一个位置,进行 $1 \rightarrow k$ 次操作的情况下 $max(len_1)$ 是多少,同时还有 $n \rightarrow 1$ 的也需要

这里很显然应该通过 dp 去解决,设定 $dp[i][j]$ 表示从 $1 \rightarrow i$ 这段区间内在保证 $i$ 被选入作为 $len_1$ 的情况下(即无论如何当前位置得是 $1$)当前的连续的 $1$ 的长度是多少,这非常简单,可以得到递推公式

$$ dp\_{i,j} = \left\{ \begin{matrix} dp\_{i-1,j} & s[i] = 1 \\\\ dp\_{i-1,j-1} & s[i] = 0 \\\\ 0 & j = 0 \end{matrix} \right. $$

然后再做一次取较大值 $dp_{i, j} = max(dp_{i, j}, dp_{i, j - 1}, dp_{i - 1, j})$,这样就可以 $O(1)$ 的方式快速得到在某个区间内,允许操作 $k$ 次的情况下,能够得到最长的 $1$ 串有多长了

然后暴力就好

AC code

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
void solve() {
int _;
cin >> _;
for (int ts = 0; ts < _; ++ts) {
int n, k;
cin >> n >> k;
string str;
str.reserve(n);
cin >> str;

vector<vector<int>> left(n);
for (auto &item : left) item.resize(k + 1);
left[0][0] = str[0] == '0' ? 0 : 1;
for (int i = 1; i <= k; ++i) left[0][i] = 1;
for (int i = 1; i < n; ++i) {
left[i][0] = str[i] == '0' ? 0 : (i == 0 ? 1 : left[i - 1][0] + 1);
for (int j = 1; j <= k; ++j) left[i][j] = str[i] == '0' ? left[i - 1][j - 1] + 1 : left[i - 1][j] + 1;
}

for (int i = 1; i < n; ++i) left[i][0] = max(left[i][0], left[i - 1][0]);
for (int i = 1; i < n; ++i) for (int j = 1; j <= k; ++j)
left[i][j] = max(max(left[i - 1][j], left[i][j]), left[i][j - 1]);

vector<vector<int>> right(n);
for (auto &item : right) item.resize(k + 1);
right[n - 1][0] = str[n - 1] == '0' ? 0 : 1;
for (int i = 1; i <= k; ++i) right[n - 1][i] = 1;
for (int i = n - 2; i >= 0; --i) {
right[i][0] = str[i] == '0' ? 0 : (i == n - 1 ? 1 : right[i + 1][0] + 1);
for (int j = 1; j <= k; ++j) right[i][j] = str[i] == '0' ? right[i + 1][j - 1] + 1 : right[i + 1][j] + 1;
}

for (int i = n - 2; i >= 0; --i) right[i][0] = max(right[i][0], right[i + 1][0]);
for (int i = n - 2; i >= 0; --i) for (int j = 1; j <= k; ++j)
right[i][j] = max(max(right[i + 1][j], right[i][j]), right[i][j - 1]);

vector<int> preS(n + 1, 0);
for (int i = 1; i <= n; ++i) preS[i] = preS[i - 1] + (str[i - 1] == '1');

vector<int> cnt(n + 1, -1);
cnt[0] = max(left[n - 1][k], right[0][k]);
for (int l = 0; l < n; ++l) {
for (int r = l; r < n && preS[r + 1] - preS[l] <= k; ++r) {
int cost = preS[r + 1] - preS[l];
int max1 = max(l == 0 ? 0 : left[l - 1][k - cost], r == n - 1 ? 0 : right[r + 1][k - cost]);
cnt[r - l + 1] = max(cnt[r - l + 1], max1);
}
}

vector<int> ans(n + 1, 0);
for (int i = 1; i <= n; ++i) for (int j = 0; j <= n; ++j)
ans[i] = max(ans[i], cnt[j] == -1 ? 0 : i * j + cnt[j]);
for (int i = 1; i <= n; ++i) cout << ans[i] << " \n"[i == n];
}
}

虽然写的挺丑,主要是不习惯写 dp,但是这段代码压根没有用到 if