CF1659 Codeforces Round #782 (Div. 2) 题解
之前说过的题解,E应该不会补了(大概)
A Red Versus Blue
题意非常简单,构造题。给定个红色气球和个蓝色气球,将它们排成一排,要求使得连续出现的最多的同色气球最少,题目保证严格小于。然而这道题我却做的令人感叹。
数据范围:
首先,可以看出这个构造出的最大的最小数量是。因为这个可以被分成份。
然后我的大脑就此停止工作了,开始进行了丑陋的模拟。主要思路是,求出来,直接先按照来分配,等到不满足时,再按照分配,最后输出剩下的。属实是三次之后的无奈之举。
void solve(){
scanf("%d%d%d", &n, &r, &b);
int mx = n / (b + 1);
while (r > b && r >= mx && b > 0){
for (int i=0;i<mx;++i) printf("R");
printf("B");
r -= mx, b--;
}
while (b > 0 && r > 0){
printf("RB");
r--, b--;
}
for (int i=0;i<b;++i){
printf("B");
}
for (int i=0;i<r;++i){
printf("R");
}
puts("");
// printf("!! %d %d\n", r, b);
}
题解是这么解决的:令,。然后自然构造两类字符串:;。代码也非常的简洁漂亮,不禁让我想起了之前一个A题的扫地机器人dalao的绝妙模拟。
这样,红色气球的数量即为。
int n, r, b;
void solve(){
scanf("%d%d%d", &n, &r, &b);
int p = r / (b + 1), q = r % (b + 1);
for (int i=0;i<q;++i){
cout << string(p + 1, 'R') << "B";
}
for (int i=q;i<b;++i){
cout << string(p, 'R') << "B";
}
cout << string(p, 'R') << endl;
}
B Bit Flipping
有一个长度为的01字符串,必须进行确切的次操作。对于每次操作,我们都要选择一个位置,使得的值不变,字符串中其他位置的值反转。需要输出在次操作后,字典序最大的字符串,并输出会在每个位置执行操作多少次。
数据范围:
我们可以推知一些对解题必要的结论。首先,对于一个给定的bit,当他不被选择的时候,就会反转。并且,如果它被选择2次,那么就相当于没有被选择。这样的话,只可能有1,或者偶数两种取值。
而且,第个bit将会被反转次。
将集合划分为如下情况:
- 是偶数,,则
- 是偶数,,则
- 是奇数,,则
- 是奇数,,则
这样从左向右执行,直到用光步操作,就可以将变为形式的字符串,字典序最大。
若执行完毕还有剩余,显然将步数都加到最后的时,字典序最大。
void solve(){
cin >> n >> m;
cin >> s;
init();
int tmp = m;
for (int i=0;i<n;++i){
if (i == n - 1) r[i] = tmp;
else if (tmp){
if ((s[i] - '0') ^ (m & 1) == 0)
r[i] = 1, --tmp;
}
if ((s[i] - '0') ^ ((m - r[i]) & 1)){
ans += "1";
}
else ans += "0";
}
cout << ans << endl;
for (int i=0;i<n;++i){
cout << r[i] << " \n"[i==n-1];
}
}
C Line Empire
大水题,自己动手做一下就会发现,求出以第个地方为都城,比较最小值,就是答案。
ll n, a, b;
const int N = 2e5 + 5;
int x[N];
ll p[N];
void solve(){
scanf("%lld%lld%lld", &n, &a, &b);
for (int i=1;i<=n;++i){
scanf("%d", &x[i]);
}
for (int i=1;i<=n;++i){
p[i] = p[i - 1] + x[i];
}
ll mn = 3e18, cur;
for (int i=0;i<=n;++i){
cur = (a + b) * x[i] + b * ((p[n] - p[i]) - (n - i) * x[i]);
mn = min(mn, cur);
}
printf("%lld\n", mn);
}
D Reverse Sort Sum
对于01数组,设函数返回结果为对的前个元素排序后的结果。定义数组。
现给定数组,确定满足条件的。
数据范围:
通过观察,我们可以发现,中1的数量。因为每个1都会在中起作用1次。
再看,如果我们从后往前处理,那么可以发现的值是可以方便求出来的。
- ,则
- ,则
现在来看例1:
4
2 4 2 4
我们可以得到:
A=[1,1,0,1];
B1=[1,1,0,1];
B2=[1,1,0,1];
B3=[0,1,1,1];
B4=[0,1,1,1];
由于,我们可以清楚地得到,那么删除的影响,就可以得到新的。相同的,我们可以判断出
通过这样处理即可得到答案。区间减1,用树状数组就可以轻松实现,复杂度为,但这不是必要的。我们可以通过一个类似滑动窗口的东西去确定影响何时消失。
#include <bits/stdc++.h>
using namespace std;
#define ll long long
int n;
const int N = 2e5 + 5;
int c[N], a[N], t[N];
void solve(){
scanf("%d", &n);
ll k = 0, L, suf = 0;
for (int i=1;i<=n;++i){
scanf("%d", &c[i]);
a[i] = 0, t[i] = n;
k += c[i];
}
k /= n, L = n - k + 1;
for (int i=n;i>=1&&L<=i;--i){
int cur = c[i] - (t[i] - i);
if (cur == i) a[i] = 1;
else if (cur == 1){
L--;
t[L] = i - 1;
}
}
for (int i=1;i<=n;++i){
printf(i==n?"%d\n":"%d ", a[i]);
}
}
int main(void){
int T;
scanf("%d", &T);
while (T--){
solve();
}
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· Manus爆火,是硬核还是营销?
· 终于写完轮子一部分:tcp代理 了,记录一下
· 别再用vector<bool>了!Google高级工程师:这可能是STL最大的设计失误
· 单元测试从入门到精通