Codeforces Round 913 (Div. 3) (A-G)

2023-12-09 02:44
文章标签 codeforces round div 913

本文主要是介绍Codeforces Round 913 (Div. 3) (A-G),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

后天就是 I C P C ICPC ICPC杭州站了,今天把之前做的 d i v 3 div3 div3题补一下,打完这场杭州站这赛季除了 E C F i n a l EC\,\,Final ECFinal就结束了,以后应该要多打 c f cf cf比赛练习保持手感,争取下赛季冲一下金牌。
感觉这个 d i v 3 div3 div3的难度还不错,正常状态应该能做到差一题 A K AK AK,思维含量还没有太高,适合我这种fw选手。

A. Rook

题面
题意:一个空的国际象棋棋盘,给你车的初始位置,问你挪动一步这个车可以到达哪些位置。
S o l u t i o n : Solution: Solution:按行按列输出即可,注意初始的点不要输出。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 1e5+10;
const ll mod = 998244353;
inline void read(int &x){int s=0,w=1;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}while(ch>='0'&&ch<='9'){s=(s<<3)+(s<<1)+(ch&15);ch=getchar();}x=s*w;
}
int n,t;
char chr,ch;
void solve(){chr=getchar();while(chr<'a'||chr>'z')chr=getchar();ch=getchar();while(ch<'0'||ch>'9')ch=getchar();n=ch-'0';for(int i=1;i<=8;i++){if(i==n)continue;printf("%c%d\n",chr,i);}for(int i=1;i<=8;i++){if('a'+i-1==chr)continue;printf("%c%d\n",'a'+i-1,n);}
}
int main(){read(t);while(t--)solve();return 0;
}

B. YetnotherrokenKeoard

题面
题意:一个奇怪的键盘,按小写 b b b是删掉目前已经打的最后一个小写字母,按大写 B B B是删掉目前已经打的最后一个大写字母(如果没有则不删,打 b / B b/B b/B不会打出来字符只会删掉字符),给出键盘按下的顺序,求最后打出的内容。
S o l u t i o n Solution Solution:个人做法是从头开始记录目前最后一个小写、大写字母的位置,然后遇到 b / B b/B b/B就删除那个位置的字符,并更新目前最后一个小写、大写字母的位置,写的比较麻烦。赛后看题解发现倒序维护 b / B b/B b/B的数量,遇到大小写直接删除是最快的最方便的。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 1e6+10;
const ll mod = 998244353;
inline void read(int &x){int s=0,w=1;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}while(ch>='0'&&ch<='9'){s=(s<<3)+(s<<1)+(ch&15);ch=getchar();}x=s*w;
}
int n,t,len,l[N],L[N],p,P,lst[N],LST[N];
char s[N];
void solve(){scanf("%s",s);len=strlen(s);p=P=0;lst[0]=LST[0]=-1;for(int i=0;i<len;i++){if(s[i]>='A'&&s[i]<='Z'){if(s[i]=='B'){if(P){s[L[P]]=0;P--;}s[i]=0;}else L[++P]=i;}else{if(s[i]=='b'){if(p){s[l[p]]=0;p--;}s[i]=0;}else l[++p]=i;}}for(int i=0;i<len;i++){if(s[i]!=0)putchar(s[i]);}puts("");
}
int main(){read(t);while(t--)solve();return 0;
}

C. Removal of Unattractive Pairs

题面
题意:对于一个字符串可以进行的操作是删掉相邻两个不相同的字符,问最后最少剩下几个字符。
S o l u t i o n Solution Solution:之前做过类似结论的题,对于出现最多的字符的出现次数 t t t,如果 t ≤ ⌊ l e n 2 ⌋ t\le\lfloor\frac{len}{2}\rfloor t2len,则不会剩下字符,否则剩下字符数为 t − ( t − ⌊ l e n 2 ⌋ ) t-(t-\lfloor\frac{len}{2}\rfloor) t(t2len⌋)

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 2e5+10;
const ll mod = 998244353;
inline void read(int &x){int s=0,w=1;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}while(ch>='0'&&ch<='9'){s=(s<<3)+(s<<1)+(ch&15);ch=getchar();}x=s*w;
}
int n,t;
int cnt[30];
char s[N];
void solve(){read(n);for(int i=0;i<=26;i++)cnt[i]=0;for(int i=1;i<=n;i++){s[i]=getchar();while(s[i]<'a'||s[i]>'z')s[i]=getchar();cnt[s[i]-'a']++;}int m=0;for(int i=0;i<=26;i++){if(cnt[i]>m)m=cnt[i];}if(m<=n-m){if(n&1)puts("1");else puts("0");}else printf("%d\n",m-n+m);
}
int main(){read(t);while(t--)solve();return 0;
}

D. Jumping Through Segments

题面
题意:一个坐标轴上有 n n n条线段,从原点 0 0 0开始移动 n n n次,每次移动到第 i i i条线段上,这些移动的最大距离 ≤ k \le k k,求 k k k的最小值。
S o l u t i o n Solution Solution:二分 k k k,判断 k k k是否可行我们通过一个贪心的思想来模拟。
首先对于当前位置 n o w now now,判断其在目标线段的左侧还是右侧,如果在左侧则要向左移动,在右侧则要向右移动,这两种情况是等价的我们以向右移动为例子。
如果向右移动到目标线段的左侧,发现还可以继续移动,这时我们就维护一个 l , r l,r l,r,代表目前可以向左、右走的多余距离。例如上述情况我们的 l = 0 , r = l=0,r= l=0,r=剩余的移动距离。
每当我们移动距离 k k k发现仍然到不了目标线段时,此时就要用到这个多余距离来判断能否通过上一步剩余的距离来使我们移动到目标线段。
对于当前位置正好在目标线段的情况,向左向右的多余距离分别为该位置到左右端点的距离。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 2e5+10;
const ll mod = 998244353;
inline void read(int &x){int s=0,w=1;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}while(ch>='0'&&ch<='9'){s=(s<<3)+(s<<1)+(ch&15);ch=getchar();}x=s*w;
}
int n,t,l[N],r[N];
bool check(int x){int now=0,suml=0,sumr=0,movl=0,movr=0;for(int i=1;i<=n;i++){int cnt=0;if(now<l[i]){cnt=l[i]-now;movr=min(r[i]-l[i],x-cnt);if(movr<0){if(sumr+movr<0)return false;}sumr+=movr,sumr=min(sumr,r[i]-l[i]),suml=0;now=l[i];}else if(now>r[i]){cnt=now-r[i];movl=min(r[i]-l[i],x-cnt);if(movl<0){if(suml+movl<0)return false;}suml+=movl,suml=min(suml,r[i]-l[i]),sumr=0;now=r[i];}else{suml=min(x+suml,now-l[i]),sumr=min(x+sumr,r[i]-now);}}return true;
}
void solve(){read(n);for(int i=1;i<=n;i++)read(l[i]),read(r[i]);int le=0,ri=1e9,mid=0;while(le<ri){int mid=le+ri>>1;if(check(mid))ri=mid;else le=mid+1;}printf("%d\n",ri);
}
int main(){read(t);while(t--)solve();return 0;
}

E. Good Triples

题面
题意:给你 n n n 问你有多少组 ( a , b , c ) (a,b,c) (a,b,c)使得 a + b + c = n a+b+c=n a+b+c=n 并且 a , b , c a,b,c a,b,c的每个位数的和等于 n n n的位数的和。
S o l u t i o n : Solution: Solution:很显然每一位不能有进位,以为数字和相等,所以对每一位都是独立考虑,各个位之间的方案数乘起来即是答案。
对于每一位的答案,赛时是通过观察样例前几个得到的答案,其实推也很好推,不想推一个一个写也写出来了,答案就是 0 + 1 + ⋯ + i 0+1+\dots+i 0+1++i,对于这一位为 i i i

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 2e5+10;
const ll mod = 998244353;
inline void read(int &x){int s=0,w=1;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}while(ch>='0'&&ch<='9'){s=(s<<3)+(s<<1)+(ch&15);ch=getchar();}x=s*w;
}
int n,t,a[]={1,3,6,10,15,21,28,36,45,55};
void solve(){read(n);ll s=1;while(n){s*=1ll*a[n%10];n/=10;}printf("%lld\n",s);
}
int main(){read(t);while(t--)solve();return 0;
}

F. Shift and Reverse

题面
题意:对于一个序列,你可以做两种操作,第一种是把最后一个放到最前面,第二种是把整个序列倒序,问最少用几次操作可以把数列变为从小到大的。不能变则输出 − 1 -1 1
S o l u t i o n Solution Solution:如果把这个序列当成循环序列,不难发现第一种操作没有改变序列,第二种操作只是改变了顺序和倒序,因此如果序列最开始不是顺序或逆序,则不能变为有序数列。
对于可以改变的序列,我们要找到序列的起始点,再分别判断先改变位置再倒序和先倒序再改变位置哪个更优,输出操作次数最少的那个即可。
赛时有点困想出来的时候还有 5 5 5分钟,再加上想出来的方法比较麻烦所以就没做这个题,实际上直接倍长数组就比较方便。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 2e5+10;
const ll mod = 998244353;
inline void read(int &x){int s=0,w=1;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}while(ch>='0'&&ch<='9'){s=(s<<3)+(s<<1)+(ch&15);ch=getchar();}x=s*w;
}
int n,t,a[N<<1];
void solve(){read(n);for(int i=1;i<=n;i++)read(a[i]),a[i+n]=a[i];int b=1,s=1,now=0,ans=n<<1;for(int i=2;i<=n*2;i++){if(a[i]>a[i-1])b++,s=1;else if(a[i]<a[i-1])s++,b=1;else s++,b++;if(s>=n||b>=n){now=i-n+1;if(s>=n)ans=min(ans,min(1+n-now+1,now));if(b>=n)ans=min(ans,min(n-now+1,2+now-1));}}if(ans!=n*2)printf("%d\n",ans);else puts("-1");
}
int main(){read(t);while(t--)solve();return 0;
}

G. Lights

题面
题意:你有 n n n个灯,有开有闭,想把这些灯全关了,但是灯有个性质是每按一次灯 i i i开关,灯 a i a_i ai的开关也会被按一次,问你最少多少次能把这些灯全关了,不能全关输出 − 1 -1 1
S o l u t i o n Solution Solution:把 i i i a i a_i ai连出一条边,就会获得一个 n n n个点 n n n条边的一张图,这个图会由若干棵基环树组成,对于每棵树我们先处理他的叶子节点,随后删掉叶子节点,最后剩下一个环,在这个环上我们每次操作都是动偶数个灯,因此如果环上有奇数个灯开着那么就无法全部关闭,输出 − 1 -1 1。对于环上有两种情况,一种是从起点开始关,一种是从起点的下一个点开始关,这个起点可以随意设置,因为环上是等价的,选出这两种情况关闭数量最少的一个进行关灯即可。

#include<bits/stdc++.h>
using namespace std;
inline void read(int &x){int s=0,w=1;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}while(ch>='0'&&ch<='9'){s=(s<<3)+(s<<1)+(ch&15);ch=getchar();}x=s*w;
}
const int N = 2e5 + 100;
int T,n,cd[N],e[N],a[N],cc[N];
void solve(){read(n);for(int i=1;i<=n;i++){char chr=getchar();while(chr!='0'&&chr!='1')chr=getchar();cc[i]=cd[i]=chr&15;}vector<set<int> > s(n+1);queue<int> p;int ans=0;for(int i=1,x;i<=n;i++){read(x);e[i]=x;s[x].insert(i);}for(int i=1;i<=n;i++)if(s[i].size()==0)p.push(i);while(!p.empty()){int i=p.front();p.pop();if(cd[i]==1){s[e[i]].erase(i);cd[i]^=1,cd[e[i]]^=1;a[++ans]=i;if(s[e[i]].size()==0)p.push(e[i]);}else{s[e[i]].erase(i);if(s[e[i]].size()==0)p.push(e[i]);}}for(int now=1;now<=n;now++){if(cd[now]){vector<int> c;int sum=1;if(cd[now])c.push_back(1);for(int i=e[now];i!=now;i=e[i]){sum++;if(cd[i])c.push_back(sum);}if(c.size()&1){puts("-1");return ;}int s1=0,s2=c[0]+sum-c[c.size()-1];for(int i=1;i<c.size();i++){if(i&1)s1+=c[i]-c[i-1];else s2+=c[i]-c[i-1];}int st=now;if(s2<=s1)st=e[now];for(int i=st;;i=e[i]){if(cd[i]){a[++ans]=i;cd[i]^=1,cd[e[i]]^=1;}if(e[i]==st)break;}}}printf("%d\n",ans);for(int i=1;i<=ans;i++)printf("%d ",a[i]);puts("");return ;
}
int main(){read(T);while(T--)solve();return 0;
}

这篇关于Codeforces Round 913 (Div. 3) (A-G)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/472195

相关文章

Codeforces Round #240 (Div. 2) E分治算法探究1

Codeforces Round #240 (Div. 2) E  http://codeforces.com/contest/415/problem/E 2^n个数,每次操作将其分成2^q份,对于每一份内部的数进行翻转(逆序),每次操作完后输出操作后新序列的逆序对数。 图一:  划分子问题。 图二: 分而治之,=>  合并 。 图三: 回溯:

Codeforces Round #261 (Div. 2)小记

A  XX注意最后输出满足条件,我也不知道为什么写的这么长。 #define X first#define Y secondvector<pair<int , int> > a ;int can(pair<int , int> c){return -1000 <= c.X && c.X <= 1000&& -1000 <= c.Y && c.Y <= 1000 ;}int m

Codeforces Beta Round #47 C凸包 (最终写法)

题意慢慢看。 typedef long long LL ;int cmp(double x){if(fabs(x) < 1e-8) return 0 ;return x > 0 ? 1 : -1 ;}struct point{double x , y ;point(){}point(double _x , double _y):x(_x) , y(_y){}point op

Codeforces Round #113 (Div. 2) B 判断多边形是否在凸包内

题目点击打开链接 凸多边形A, 多边形B, 判断B是否严格在A内。  注意AB有重点 。  将A,B上的点合在一起求凸包,如果凸包上的点是B的某个点,则B肯定不在A内。 或者说B上的某点在凸包的边上则也说明B不严格在A里面。 这个处理有个巧妙的方法,只需在求凸包的时候, <=  改成< 也就是说凸包一条边上的所有点都重复点都记录在凸包里面了。 另外不能去重点。 int

Codeforces 482B 线段树

求是否存在这样的n个数; m次操作,每次操作就是三个数 l ,r,val          a[l] & a[l+1] &......&a[r] = val 就是区间l---r上的与的值为val 。 也就是意味着区间[L , R] 每个数要执行 | val 操作  最后判断  a[l] & a[l+1] &......&a[r] 是否= val import ja

CSS实现DIV三角形

本文内容收集来自网络 #triangle-up {width: 0;height: 0;border-left: 50px solid transparent;border-right: 50px solid transparent;border-bottom: 100px solid red;} #triangle-down {width: 0;height: 0;bor

创建一个大的DIV,里面的包含两个DIV是可以自由移动

创建一个大的DIV,里面的包含两个DIV是可以自由移动 <body>         <div style="position: relative; background:#DDF8CF;line-height: 50px"> <div style="text-align: center; width: 100%;padding-top: 0px;"><h3>定&nbsp;位&nbsp;

Codeforces Round 971 (Div. 4) (A~G1)

A、B题太简单,不做解释 C 对于 x y 两个方向,每一个方向至少需要 x / k 向上取整的步数,取最大值。 由于 x 方向先移动,假如 x 方向需要的步数多于 y 方向的步数,那么最后 y 方向的那一步就不需要了,答案减 1 代码 #include <iostream>#include <algorithm>#include <vector>#include <string>

CF#271 (Div. 2) D.(dp)

D. Flowers time limit per test 1.5 seconds memory limit per test 256 megabytes input standard input output standard output 题目链接: http://codeforces.com/contest/474/problem/D We s

CF #278 (Div. 2) B.(暴力枚举+推导公式+数学构造)

B. Candy Boxes time limit per test 1 second memory limit per test 256 megabytes input standard input output standard output 题目链接: http://codeforces.com/contest/488/problem/B There