Codeforces Round 970 (Div. 3) A~F

2024-09-02 10:28
文章标签 codeforces round div 970

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

封面原图 画师青眼鏡

Codeforces Round 970 (Div. 3)

A - Sakurako’s Exam

题意

给你a个1和b个2,它们的顺序和中间的加减符号随意,问你能不能写成结果为0的算式

思路

假设全部都是正的加上去,然后变一个2会使最终结果减4,变一个1会减2,看最后能不能减到0即可

代码

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef double db;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;void solve()
{int a,b;scanf("%d%d",&a,&b);int now=a+b*2;while(b>0 and now>=4){b--;now-=4;}while(a>0 and now>=2){a--;now-=2;}if(now==0){puts("YES");}else{puts("NO");}
}int main()
{int T=1;scanf("%d",&T);while(T--){solve();}return 0;
}

B - Square or Not

题意

把一个矩阵每行收尾相接之后给你,问你能不能还原成正方形的下面这种1包0的矩阵
在这里插入图片描述

思路

如果不说是正方形还会麻烦很多,知道是正方形也就意味着边长知道了,直接判断即可

代码

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef double db;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;void solve()
{int n;scanf("%d",&n);string s;cin>>s;if((int)sqrt(n)*(int)sqrt(n)!=n){puts("No");return;}int m=(int)sqrt(n);for(int i=0;i<m;i++){if(s[i]!='1'){puts("No");return;}}for(int i=n-1;i>=n-m;i--){if (s[i] != '1'){puts("No");return;}}for(int i=m;i<n-m;i++){if((i%m==0 or i%m==m-1) and s[i]!='1'){puts("No");return;}if(i%m!=0 and i%m!=m-1 and s[i]!='0'){puts("No");return;}}puts("Yes");
}int main()
{int T=1;scanf("%d",&T);while(T--){solve();}return 0;
}

C - Longest Good Array

题意

问你能构造的满足以下条件的最长数组的长度是多少

  1. 对于a中每个元素都有 l ≤ a i ≤ r l \le a_i \le r lair
  2. 数组a严格递增
  3. 数组a中每两个数的差值也严格递增

思路

就是往后+1,+2,+3,+4,…以此类推即可,直接加应该都行,但是我还简化了一下,改成了求和公式

代码

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef double db;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;void solve()
{ll l,r;scanf("%lld%lld",&l,&r);ll del=r-l;if(del==0){puts("1");return;}ll tmp=(int)sqrt(del*2)-2;while(tmp*(tmp+1)/2<=del){tmp++;}printf("%lld\n",tmp);
}int main()
{int T=1;scanf("%d",&T);while(T--){solve();}return 0;
}

D - Sakurako’s Hobby

题意

对于某个排列 p p p 如果可以通过赋值 i = p i i=p_i i=pi 一定次数使 i i i 等于 j j j ,则樱子称整数 j j j 可以从整数 i i i 到达。
如果是 p = [ 3 , 5 , 6 , 1 , 2 , 4 ] p=[3,5,6,1,2,4] p=[3,5,6,1,2,4] ,那么,举例来说, 4 4 4 可以从 1 1 1 到达,因为: i = 1 i=1 i=1 → \rightarrow i = p 1 = 3 i=p_1=3 i=p1=3 → \rightarrow i = p 3 = 6 i=p_3=6 i=p3=6 → \rightarrow i = p 6 = 4 i=p_6=4 i=p6=4 .现在是 i = 4 i=4 i=4 ,所以 4 4 4 可以从 1 1 1 到达。
排列中的每个数字都被染成黑色或白色。
樱子将函数 F ( i ) F(i) F(i) 定义为从 i i i 可以到达的黑色整数的个数。
樱子对每个 1 ≤ i ≤ n 1\le i\le n 1in F ( i ) F(i) F(i) 都很感兴趣,但计算所有值变得非常困难,所以她请你作为她的好朋友来计算这个值。

思路

建图找环,dfs也行并查集也行,我用的是dfs

代码

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef double db;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;
const int N=200010;
int a[N],ans[N];
int n;
string s;
bool vis[N];
int cnt;
int dfs(int p)
{if(vis[p]){return cnt;}vis[p]=1;if(s[p]=='0')cnt++;ans[p]=dfs(a[p]);return cnt;
}void solve()
{scanf("%d",&n);for(int i=1;i<=n;i++)scanf("%d",&a[i]);s="";cin>>s;s=" "+s;memset(ans,-1,sizeof(ans));for(int i=1;i<=n;i++){if(ans[i]==-1){cnt=0;memset(vis,0,sizeof(vis));ans[i] = dfs(i);}}for(int i=1;i<=n;i++)printf("%d ",ans[i]);printf("\n");
}int main()
{int T=1;scanf("%d",&T);while(T--){solve();}return 0;
}

F - Sakurako’s Box

题意

求一个数组里任取两个数的乘积的期望值

思路

直接模拟计算即可,每个数都要乘一遍所有数,用乘法分配律,就是乘上所有其他数的和,前缀和维护一下就行

代码

C++

这个会WA5,精度一直调不对,后来怒而改成python直接过了

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef double db;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;
const ll mod=1e9+7;
const int N=200010;
ll a[N],presum[N];
int n;
ll qpow(ll x,ll y)
{ll res=1;while(y){if(y&1) res=res*x%mod;x=x*x%mod;y>>=1;}return res;
}void solve()
{scanf("%d",&n);presum[0]=0;for(int i=1;i<=n;i++){scanf("%lld", &a[i]);presum[i]=presum[i-1]+a[i];presum[i]%=mod;}ll p=0;for(int i=1;i<=n;i++){p+=a[i]*presum[i-1]%mod;p%=mod;}ll q=n*(n-1)/2%mod;ll ans=p*qpow(q,mod-2)%mod;printf("%lld\n",ans);
}int main()
{int T=1;scanf("%d",&T);while(T--){solve();}return 0;
}

Python

import sys
from typing import Listmod = 10**9 + 7
N = 200010
a = [0]*N
presum = [0]*Ndef qpow(x: int, y: int) -> int:res = 1while y:if y & 1:res = res * x % modx = x * x % mody >>= 1return resdef solve() -> None:n = int(sys.stdin.readline())a[1:n+1] = map(int, sys.stdin.readline().split())for i in range(1, n+1):presum[i] = (presum[i-1] + a[i])p = 0for i in range(1, n+1):p = (p + a[i]*presum[i-1])q = n*(n-1)//2ans = p*qpow(q, mod-2)ans %= modprint(ans)def main() -> None:T = int(sys.stdin.readline())for _ in range(T):solve()if __name__ == "__main__":main()

E - Alternating String

题意

樱子非常喜欢交替字符串。如果偶数位置上的字符相同,奇数位置上的字符相同,并且字符串的长度是偶数,她就会把一个由小写拉丁字母组成的字符串 s s s 称为交替字符串。
例如,字符串 "abab "和 "gg "是交替字符串,而字符串 "aba "和 "ggwp "则不是。
作为好朋友,您决定赠送这样的字符串,但您找不到。幸运的是,您可以对字符串进行两种操作:

  1. 选择索引 i i i 并删除字符串中的 i i i -th 字符,这将使字符串的长度减少 1 1 1 。此类操作的**次数不超过 1 1 1 次;
  2. 选择索引 i i i 并将 s i s_i si 替换为其他字母。
    由于时间紧迫,您需要确定使字符串成为交替字符串所需的最少操作次数。

思路

分情况讨论,如果这个字符串的长度是偶数那就不需要删,因为删完不会改变相对奇偶位置,总可以找到等价的修改方式,奇数就遍历删除的位置,其他的操作不变就行了

代码

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<utility>
#include<bitset>
#include<string>
#include<vector>
#include<cstdio>
#include<stack>
#include<queue>
#include<deque>
#include<cmath>
#include<map>
#include<set>using namespace std;
#define int long long
const int N=200010;
const int Base=255;
const int MOD=1e9+7;int pre1[N][30],pre2[N][30];signed main()
{ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);int t;cin>>t;while(t--){int n;cin>>n;string s;cin>>s;if(n%2==0){for(int i=1;i<=n;i++){for(int j=1;j<=26;j++)pre1[i][j] = pre1[i-1][j];if(i%2) pre1[i][s[i-1]-'a'+1]++;}for(int i=2;i<=n;i++){for(int j=1;j<=26;j++)pre2[i][j] = pre2[i-1][j];if(i%2==0) pre2[i][s[i-1]-'a'+1]++;}int maxvalue1=0,maxvalue2=0;for(int j=1;j<=26;j++){maxvalue1=max(maxvalue1,pre1[n][j]);maxvalue2=max(maxvalue2,pre2[n][j]);}cout<<n-maxvalue1-maxvalue2<<"\n";}else {//先看奇数位置for(int i=1;i<=n;i++){for(int j=1;j<=26;j++)pre1[i][j] = pre1[i-1][j];if(i%2) pre1[i][s[i-1]-'a'+1]++;}//先看偶数位置for(int i=2;i<=n;i++){for(int j=1;j<=26;j++)pre2[i][j] = pre2[i-1][j];if(i%2==0) pre2[i][s[i-1]-'a'+1]++;}int res=n;for(int i=1;i<=n;i++)//遍历进行删除操作的位置{int maxvalue1=0,maxvalue2=0;for(int j=1;j<=26;j++){
//					cout<<i<<" "<<j << " " << pre1[i][j] <<" " << pre2[i][j]<<"\n";maxvalue1 = max(maxvalue1 , pre1[i-1][j] + pre2[n][j] - pre2[i][j] );maxvalue2 = max(maxvalue2 , pre2[i-1][j] + pre1[n][j] - pre1[i][j] );}
//				cout<<i<<" "<<maxvalue1<<" "<<maxvalue2<<"\n";res = min(res,n-maxvalue1-maxvalue2);}cout<<res<<"\n";}}return 0;
}

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



http://www.chinasem.cn/article/1129708

相关文章

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