Codeforces Round #790 (Div. 4) A-H

A

题目

https://codeforces.com/contest/1676/problem/A

题解

思路

知识点:模拟。

照着模拟(细节加0防炸char,虽然这里没用)。

时间复杂度 \(O(1)\)

空间复杂度 \(O(1)\)

代码

#include <bits/stdc++.h>

using namespace std;

int main(){
std::ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int t;
cin>>t;
while(t--){
string s;
cin>>s;
if(0+s[0]+s[1]+s[2] == 0+s[3]+s[4]+s[5]) cout<<"YES"<<'\n';
else cout<<"NO"<<'\n';
} return 0;
}

B

题目

https://codeforces.com/contest/1676/problem/B

题解

思路

知识点:贪心。

所有数减去最小的加在一起就行。

时间复杂度 \(O(n)\)

空间复杂度 \(O(n)\)

代码

#include <bits/stdc++.h>

using namespace std;

int a[57];

int main(){
std::ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int t;
cin>>t;
while(t--){
int n;
cin>>n;
int ans = 0,mincnt = 1e8;
for(int i = 0;i<n;i++){
cin>>a[i];
mincnt = min(mincnt,a[i]);
ans += a[i];
}
ans -= n * mincnt;
cout<<ans<<'\n';
} return 0;
}

C

题目

https://codeforces.com/contest/1676/problem/C

题解

思路

知识点:暴力。

(看错题浪费20分钟写成求全部字符串变成一样的最小次数。。。)

直接暴力求最小值就行。

时间复杂度 \(O(n^2m)\)

空间复杂度 \(O(nm)\)

代码

#include <bits/stdc++.h>

using namespace std;

string s[57];

int main(){
std::ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int t;
cin>>t;
while(t--){
int n,m;
cin>>n>>m;
for(int i = 0;i<n;i++) cin>>s[i];
int ans = ~(1<<31);
for(int i = 0;i<n;i++){
for(int j = i+1;j<n;j++){
int sum = 0;
for(int k = 0;k<m;k++){
sum += abs(s[i][k] - s[j][k]);
}
ans = min(ans,sum);
}
}
cout<<ans<<'\n';
}
return 0;
}

D

题目

https://codeforces.com/problemset/problem/1676/D

题解

思路

知识点:暴力

暴力枚举最大值,注意边界。

时间复杂度 \(O(n^2m)\)

空间复杂度 \(O(nm)\)

代码

#include <bits/stdc++.h>

using namespace std;

int a[207][207];

int main(){
std::ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int t;
cin>>t;
while(t--){
int n,m;
cin>>n>>m;
for(int i = 0;i<n;i++)
for(int j = 0;j<m;j++)
cin>>a[i][j];
int ans = 0;
for(int i = 0;i<n;i++){
for(int j = 0;j<m;j++){
int sum = 0;
for(int k = 0;k<n;k++){
if(0<= j-(i-k) && j-(i-k) <m)sum += a[k][j-(i-k)];
if(0<= j+(i-k) && j+(i-k) <m && j+(i-k) != j-(i-k))sum += a[k][j+(i-k)];
}
ans = max(ans,sum);
}
}
cout<<ans<<'\n';
}
return 0;
}

E

题目

https://codeforces.com/problemset/problem/1676/E

题解

思路

知识点:前缀和,二分查找,贪心。

每次都选最大的即可最少消耗到达目标 \(x\) ,考虑预处理从大到小的前缀和,二分查找第一个大于等于 \(x\) 的和的下标即可。

注意 \(lower\_bound\) 和 \(upper\_bound\) 函数的用法,前者查找大于等于的第一个下标,后者查找大于的第一个下标。并且只可查找“升序”序列,当然这个升序可以自定义,比如用 \(greater\) 把大于定义成升序,即可查找从大到小的序列。

时间复杂度 \(O(nlogn)\)

空间复杂度 \(O(n)\)

代码

#include <bits/stdc++.h>

using namespace std;

int a[150007];

bool cmp(int a,int b){
return a>b;
} int main(){
std::ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int t;
cin>>t;
while(t--){
int n,q;
cin>>n>>q;
for(int i = 1;i<=n;i++) cin>>a[i];
sort(a+1,a+1+n,cmp);
for(int i = 1;i<=n;i++) a[i] += a[i-1];
while(q--){
int x;
cin>>x;
int ans = lower_bound(a+1,a+1+n,x) - a;
cout<<(ans<=n?ans:-1)<<'\n';
}
}
return 0;
}

F

题目

https://codeforces.com/problemset/problem/1676/F

题解

思路

方法1

知识点:贪心,双指针,模拟。

遍历排序后的数组 \(a\) ,记录相同数字个数 \(sum\),如果 \(sum \geq k\) ,则说明这个数字合法,记入临时右值 \(r\) 中,并重置 $sum = 1 $ ,即记录了下一段第一个数字的一次 。

直到某个数字 \(a[i-1]\) 的 \(sum<k\) 或者与后一个数字 \(a[i]\) 差大于 \(1\) ,即 \(a[i] - a[i-1] > 1\) ,说明 \(a[i-1]\) 这个数字在目前段中是最后一个可能合法数字。此时有两种情况,如果 \(sum \geq k\) 则说明数字 \(a[i-1]\) 是合法的更新右值 \(r\) ,否则不合法。之后更新答案,并更新左值为 \(a[i]\) 以及 \(sum = 1\),即新段开始。

遍历结束就可以得到答案。注意初始化方便判断无解。

小细节,如果直接遍历,最后一段合法段可能没法判断,在循环外额外加个判断太丑,可以在末尾加一个不合法的数,循环范围调大一个数字,这样就可以用最后一个不合法数字中断遍历进行判断,而且不合法数字不会参与判断。

时间复杂度 \(O(nlogn)\)

空间复杂度 \(O(n)\)

方法2

知识点:贪心,离散化,双指针,模拟。

用 \(map\) 做数字和次数的映射,可以将区间压缩到数字一个点,再进行遍历即可。思路一样,不合法次数的数字直接跳过,合法数字直接更新右值为这个数字,若这个数字与上一个区间的右值差大于 \(1\) 则更新左值为此数字。每次都更新答案可以少在循环外加一个判断去判断最后一段。

时间复杂度 \(O(nlogn)\)

空间复杂度 \(O(n)\)

两种方法,前者写起来麻烦但常数小,后者写起来方便但常数大容易被卡。

代码

方法1
///直接遍历,非常快
#include <bits/stdc++.h> using namespace std; int a[200007]; int main(){
std::ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int t;
cin>>t;
while(t--){
int n,k;
cin>>n>>k;
for(int i = 0;i<n;i++){
cin>>a[i];
}
sort(a,a+n);
a[n] = a[n-1] + 2;///小细节,防止最后一段不判断,在后面加个新段就行
int lans = 1,rans = 0,l = a[0],r = 0,sum = 1;
///lans最大区间的左值,rans最大区间的右值,l左值,r右值,sum计数
for(int i = 1;i<=n;i++){///这里需要到n,因为后面有个用来终止的段,不怕越界
if(a[i] == a[i-1]) sum++;
else{
if(sum<k || a[i] - a[i-1] > 1){///此段终止
if(sum>=k) r = a[i-1];///若合法,则更新最大右值
if(rans - lans < r - l) rans = r,lans = l;///更新答案
l = a[i];///更新左值
}
else{///还可以继续增加
r = a[i-1];///更新最大右值
}
sum = 1;///重置计数
}
}
if(!rans) cout<<-1<<'\n';
else cout<<lans<<' '<<rans<<'\n';
}
return 0;
}
方法2
///map,好写但常数大
#include <bits/stdc++.h> using namespace std; int main(){
std::ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int t;
cin>>t;
while(t--){
map<int,int> cnt;
int n,k;
cin>>n>>k;
for(int i = 0;i<n;i++){
int tmp;
cin>>tmp;
cnt[tmp]++;
}
int lans = 0,rans = -1;///保证r-l的合法也能更新,同时rans能用来判断无解
int l = 0,r = -1;///r保证第一段合法开始l就会被更新,于是l随意
for(auto [i,j]:cnt){///c++17才有这个用法。。。不然老老实实单变量
if(j < k) continue;
if(i - r > 1) l = i;
r = i;
if(rans - lans < r - l){///每次都更新就可以避免最后一段没判断
rans = r;
lans = l;
}
}
if(!~rans) cout<<-1<<'\n';
else cout<<lans<<' '<<rans<<'\n';
}
return 0;
}

G

题目

https://codeforces.com/problemset/problem/1676/G

题解

思路

方法1

知识点:DFS,图论,DP。

用图的方式存树,可以构建一个双向的树,方便从根搜索孩子。

深搜,自上而下搜索子树,自底向上累加子树颜色和,先搜索子树后累加实现。

时间复杂度 \(O(n)\)

空间复杂度 \(O(n^2)\)

方法2

知识点:拓扑排序,图论,DP。

用父亲表示法建树,同时建立节点度的数组。

拓扑排序,自度为0的节点(叶子节点,无子树)开始向上层搜索,进行累加颜色和,一个节点度变为0证明子树颜色和完毕,可以把其放入队列,向其父节点遍历。

时间复杂度 \(O(n)\)

空间复杂度 \(O(n)\)

感觉方法1的常数比方法2小。

代码

方法1
///深搜,自底向上累加子树颜色和,先搜后加实现
#include <bits/stdc++.h> using namespace std; int c[4007];
vector<int> g[4007]; void dfs(int u,int fa){
for(int i = 0;i<g[u].size();i++){
if(g[u][i] == fa) continue;///不能以下犯上233
dfs(g[u][i],u);///先把子树的颜色数解决
c[u] += c[g[u][i]];///当前节点的为根的子树颜色数等于自己颜色加所有孩子子树的颜色数
}
} int main(){
std::ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int t;
cin>>t;
while(t--){
int n;
cin>>n;
for(int i = 1;i<=n;i++) g[i].clear(),c[i] = 0;
for(int i = 2;i<=n;i++){
int fa;
cin>>fa;
g[i].push_back(fa);
g[fa].push_back(i);
}
for(int i = 1;i<=n;i++){
char col;
cin>>col;
if(col == 'W') c[i] = 1;
else if(col == 'B') c[i] = -1;///黑色-1,白色1,平衡子树总和为0
} dfs(1,-1);
int ans = 0;
for(int i = 1;i<=n;i++){
ans += c[i] == 0;
}
cout<<ans<<'\n';
}
return 0;
}
方法2
///拓扑排序,自底向上累加子树颜色和,先度为0(叶子)的节点向上累加实现
#include <bits/stdc++.h> using namespace std; int c[4007],deg[4007];
int fa[4007]; void toposort(int n){
queue<int> q;
for(int i = 1;i<=n;i++){
if(!deg[i]) q.push(i);
}
while(!q.empty()){
int cur = q.front();
q.pop();
c[fa[cur]] += c[cur];
deg[fa[cur]]--;
if(!deg[fa[cur]]) q.push(fa[cur]);
}
} int main(){
std::ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int t;
cin>>t;
while(t--){
int n;
cin>>n;
for(int i = 1;i<=n;i++) deg[i] = c[i] = 0;
for(int i = 2;i<=n;i++){
cin>>fa[i];
deg[fa[i]]++;
}
for(int i = 1;i<=n;i++){
char col;
cin>>col;
if(col == 'W') c[i] = 1;
else if(col == 'B') c[i] = -1;///黑色-1,白色1,平衡子树总和为0
} toposort(n);
int ans = 0;
for(int i = 1;i<=n;i++){
ans += c[i] == 0;
}
cout<<ans<<'\n';
}
return 0;
}

H

题目

https://codeforces.com/problemset/problem/1676/H1

https://codeforces.com/problemset/problem/1676/H2

题解

思路

方法1

知识点:分治,归并排序。

通过肉眼观察法得到最大交点数实际上就是逆序数变种,区别就是等于的也要算,用归并排序改一下即可。

时间复杂度 \(O(nlogn)\)

空间复杂度 \(O(n)\)

方法2

知识点:树状数组。

这块还不会,改天再学,先贴大佬代码。

时间复杂度 \(O(?)\)

空间复杂度 \(O(?)\)

代码

方法1
///归并排序
#include <bits/stdc++.h> using namespace std; long long cnt = 0;
int a[200000+7],b[200000+7]; void merge_sort(int l,int r){
if(l >= r) return;///必须>=,因为mid+1可能会大于
int mid = (l+r)/2;
merge_sort(l,mid);
merge_sort(mid+1,r);
int i = l,j = mid+1,k = l;
while(i<=mid && j<=r){
if(a[i]<a[j])///改了这里
b[k++] = a[i++];
else
b[k++] = a[j++],cnt+=mid-i+1;///表示从i到mid的数字都大于a[j]都要算一遍
}
while(i<=mid)
b[k++] = a[i++];
while(j<=r)
b[k++] = a[j++];
memcpy(a+l,b+l,(r-l+1)*sizeof(int));
} int main(){
std::ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
int t;
cin>>t;
while(t--){
int n;
cin>>n;
for(int i = 0;i<n;i++)
cin>>a[i];
cnt = 0;
merge_sort(0,n-1);
cout<<cnt<<'\n';
}
return 0;
}
方法2
///树状数组
#include <bits/stdc++.h> using namespace std; template <typename T>
struct Fenwick{
const int n;
vector<T> a;
Fenwick(int n):n(n),a(n){}
void add(int x,T v) {
for (int i = x+1;i<=n;i += i&-i){
a[i-1] += v;
}
}
T sum(int x) {
T ans = 0;
for (int i = x;i>0;i -= i&-i){
ans += a[i-1];
}
return ans;
}
T rangeSum(int l,int r){
return sum(r) - sum(l);
}
}; void solve(){
int n;
cin >> n;
vector<int> a(n);
for (int i = 0;i<n;i++) {
cin>>a[i];
a[i]--;
}
Fenwick<int> fen(n);
long long ans = 0;
for (int i = n-1;i>=0;i--) {
ans += fen.sum(a[i]+1);
fen.add(a[i],1);
}
cout<<ans<<"\n";
} int main(){
std::ios::sync_with_stdio(false),cin.tie(0),cout.tie(0); int t;
cin>>t; while(t--){
solve();
} return 0;
}

Codeforces Round #790 (Div. 4) A-H的更多相关文章

  1. Codeforces Round #371 (Div. 1)

    A: 题目大意: 在一个multiset中要求支持3种操作: 1.增加一个数 2.删去一个数 3.给出一个01序列,问multiset中有多少这样的数,把它的十进制表示中的奇数改成1,偶数改成0后和给 ...

  2. Codeforces Round #160 (Div. 1) 题解【ABCD】

    Codeforces Round #160 (Div. 1) A - Maxim and Discounts 题意 给你n个折扣,m个物品,每个折扣都可以使用无限次,每次你使用第i个折扣的时候,你必须 ...

  3. Codeforces Round #383 (Div. 2) 题解【ABCDE】

    Codeforces Round #383 (Div. 2) A. Arpa's hard exam and Mehrdad's naive cheat 题意 求1378^n mod 10 题解 直接 ...

  4. Codeforces Round #271 (Div. 2)题解【ABCDEF】

    Codeforces Round #271 (Div. 2) A - Keyboard 题意 给你一个字符串,问你这个字符串在键盘的位置往左边挪一位,或者往右边挪一位字符,这个字符串是什么样子 题解 ...

  5. Codeforces Round #177 (Div. 1) 题解【ABCD】

    Codeforces Round #177 (Div. 1) A. Polo the Penguin and Strings 题意 让你构造一个长度为n的串,且里面恰好包含k个不同字符,让你构造的字符 ...

  6. Codeforces Round #182 (Div. 1)题解【ABCD】

    Codeforces Round #182 (Div. 1)题解 A题:Yaroslav and Sequence1 题意: 给你\(2*n+1\)个元素,你每次可以进行无数种操作,每次操作必须选择其 ...

  7. 线段树 Codeforces Round #197 (Div. 2) D. Xenia and Bit Operations

    题目传送门 /* 线段树的单点更新:有一个交叉更新,若rank=1,or:rank=0,xor 详细解释:http://www.xuebuyuan.com/1154895.html */ #inclu ...

  8. Codeforces Round #378 (Div. 2) D题(data structure)解题报告

    题目地址 先简单的总结一下这次CF,前两道题非常的水,可是第一题又是因为自己想的不够周到而被Hack了一次(或许也应该感谢这个hack我的人,使我没有最后在赛后测试中WA).做到C题时看到题目情况非常 ...

  9. Codeforces Round #373 (Div. 2)A B

    Codeforces Round #373 (Div. 2) A. Vitya in the Countryside 这回做的好差啊,a想不到被hack的数据,b又没有想到正确的思维 = = [题目链 ...

随机推荐

  1. 在 WinForms 项目中使用全局快捷键

    借助于全局快捷键,用户可以在任何地方操控程序,触发对应的功能.但 WinForms 框架并没有提供全局快捷键的功能.想要实现全局快捷键需要跟 Windows API 打交道.本文就交你如何使用 Win ...

  2. 用python爬虫,对12306网站进行模拟登陆

    from selenium import webdriver from time import sleep from PIL import Image from selenium.webdriver ...

  3. IDEA SpringBoot-Mybatis-plus 实现增删改查(CRUD)

    上一篇: IDEA SpringBoot-Mybatis实现增删改查(CRUD) 下一篇:Intellij IDEA 高效使用教程 (插件,实用技巧) 最好用的idea插件大全 一.前言 Mybati ...

  4. SSM框架中返回的是字符串还是页面跳转的问题

    如果你在控制器前的注解是@RestController的话,将返回controller方法指定的String值,@RestController注解相当于@ResponseBody和@Controlle ...

  5. 自学java的困难

    在自学的一些基础阶段,倒是没什么太大的问题,但是在想搞一个项目的时候,就显得手足无措了.因为,很多博主讲的的那些,都行需要一定的条件,比如前端框架,数据库的数据等等. 简单一点的SSM框架整合相对简单 ...

  6. [笔记] prufer 序列

    什么是 prufer 序列 是可以和 \(n\) 个有标号节点的无根树一一对应的长度为 \(n-2\) 的序列. 一般来说是用于和树相关的组合计数问题,但是可能会出现一些变形,所以除了要了解一些性质, ...

  7. Spring 源码(9)Spring Bean的创建过程的前期准备

    回顾总结 到目前为止,Spring源码中AbstractApplicationContext#refresh方法的已经解读到第11个方法finishBeanFactoryInitialization, ...

  8. .NET混合开发解决方案11 WebView2加载的网页中JS调用C#方法

    系列目录     [已更新最新开发文章,点击查看详细] WebView2控件应用详解系列博客 .NET桌面程序集成Web网页开发的十种解决方案 .NET混合开发解决方案1 WebView2简介 .NE ...

  9. django-rest-framework 基础三 认证、权限和频率

    django-rest-framework 基础三 认证.权限和频率 目录 django-rest-framework 基础三 认证.权限和频率 1. 认证 1.1 登录接口 1.2 认证 2. 权限 ...

  10. 385. Mini Parser - LeetCode

    Question 385. Mini Parser Solution 分析:用NI(count,list)来表示NestedInteger,则解析字符串[123,[456,[789]]]过程如下: # ...