Codeforces Round 944 (Div. 4)(A,B,C,D,E,F,G,H)

比赛链接

这场不难, G G G H H H 比较有意思。 G G G 题需要一定的二进制和数据结构的知识, H H H 题是个 2 − s a t 2-sat 2sat 的题,算法名字吓人但是其实很简单,题目本身也很板,建议趁机学习一波。


A. My First Sorting Problem

题意 :

给你两个整数 x x x y y y

输出两个整数: x x x y y y 的最小值,以及 x x x y y y 的最大值。

思路:

签到

code:

#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;int T,a,b;int main(){cin>>T;while(T--){cin>>a>>b;if(a>b)swap(a,b);cout<<a<<" "<<b<<endl;}return 0;
}

B. Different String

题意:

给你一个由小写英文字母组成的字符串 s s s

s s s 中的字符重新排列,组成一个新的字符串 r r r ,这个字符串不等于 s s s ,或者报告说这是不可能的。

思路:

如果不可以变成另一个字符串,那么这个字符串一定是由同一个字符组成的。要不然我们就可以选择两个不同的字符交换位置,这样得到的字符串就是不一样的。

我们不妨先选择第一个字符,然后向后找与它不同的字符串,然后交换一下即可。查找不同字符可以使用 string 的成员函数 int string::find_first_not_of(str,pos),它的作用是从第 p o s pos pos 位置开始从前到后寻找第一个不在 s t r str str 中出现的字符,并返回它的下标位置,如果找不到则返回 string::npos

和它作用类似的还有 find_first_of()find_last_of()find_last_not_of()。成员函数 find(str,pos) 的作用是找 str,是子串匹配,而不是找字符。

code:

#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;int T;
string s;int main(){cin>>T;while(T--){cin>>s;int idx=s.find_first_not_of(s[0],1);if(idx==string::npos)puts("NO");else {puts("YES");swap(s[0],s[idx]);cout<<s<<endl;}}return 0;
}

C. Clock and Strings

题意:

如下图所示,有一个时钟,上面按顺时针顺序标有 1 1 1 12 12 12 的数字。

在本例中, ( a , b , c , d ) = ( 2 , 9 , 10 , 6 ) (a,b,c,d)=(2,9,10,6) (a,b,c,d)=(2,9,10,6) 和字符串相交。

爱丽丝和鲍勃有四个不同的整数 a a a b b b c c c d d d ,且不大于 12 12 12 。爱丽丝用红色字符串连接 a a a b b b ,鲍勃用蓝色字符串连接 c c c d d d 。这两条线相交吗?(字符串是直线段)。

思路:

因为题目说了 a , b , c , d a,b,c,d a,b,c,d 互不重复,我们就不考虑相等的特殊情况了。因为 a , b a,b a,b 没有先后顺序,所以不妨令 a < b a<b a<b,方便讨论,同理 c < d c<d c<d

手玩一下发现只有 a < c < b < d a<c<b<d a<c<b<d 或者 c < a < d < b c<a<d<b c<a<d<b 两种情况下会相交。判断一下即可。

code:

#include <iostream>
#include <cstdio>
using namespace std;int T,a,b,c,d;int main(){cin>>T;while(T--){cin>>a>>b>>c>>d;if(a>b)swap(a,b);if(c>d)swap(c,d);puts(((a<c && c<b && b<d) || (c<a && a<d && d<b))?"YES":"NO");}return 0;
}

D. Binary Cut

题意:

给你一个二进制字符串 † ^{\dagger} 。请找出最少需要切割成多少个片段,以便将得到的片段重新排列成一个有序的二进制字符串。

请注意

  • 每个字符必须正好位于其中一个片段中;
  • 片段必须是原始字符串的连续子串;
  • 在重新排列时必须使用所有片段。

† ^{\dagger} 二进制字符串是由字符 0 \texttt{0} 0 1 \texttt{1} 1 组成的字符串。排序后的二进制字符串是指所有字符 0 \texttt{0} 0 都位于所有字符 1 \texttt{1} 1 之前的二进制字符串。

思路:

因为 0 0 0 在前, 1 1 1 在后,因此我们在切割的时候,除了一段可以是前面一段 0 0 0 后面一段 1 1 1 以外,其他的段必须切成全 0 0 0 或全 1 1 1

考虑前面一段 0 0 0 后面一段 1 1 1 这种怎么切会比较复杂,我们可以把它看成是一个全 0 0 0 段和一个全 1 1 1 段,只不过中间接起来了。这样我们只需要先把原片段切成全 0 0 0 段和全 1 1 1 段就行了,显然我们在 0 → 1 0\rightarrow 1 01 1 → 0 1\rightarrow 0 10 变化的中间切就行了,数一下。

然后我们再把一个全 0 0 0 段和一个全 1 1 1 段中间接起来,当作没切过,在答案上减一。不过我们不能保证原片段一定存在前面一段 0 0 0 后面一段 1 1 1 的情况,所以我们再看一下有没有 0 → 1 0\rightarrow 1 01 的变化,没有就说明不能拼起来,也就不能给答案减一。

code:

#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;int T;
string s;int main(){cin>>T;while(T--){cin>>s;int n=s.length(),cnt=1;bool flag=false;//0->1for(int i=1;i<n;i++){if(s[i-1]=='0' && s[i]=='1')flag=true;if(s[i]!=s[i-1])cnt++;}cout<<cnt-flag<<endl;}return 0;
}

E. Find the Car

题意:

Timur 坐在一辆汽车上,从点 0 0 0 沿数线行驶到点 n n n 。在第 0 0 0 分钟,汽车从第 0 0 0 点开始行驶。

0 , a 1 , a 2 , … , a k 0, a_1, a_2, \dots, a_k 0,a1,a2,,ak 点的直线上有 k + 1 k+1 k+1 个标志,帖木儿知道汽车将分别在 0 , b 1 , b 2 , … , b k 0, b_1, b_2, \dots, b_k 0,b1,b2,,bk 分钟到达那里。序列 a a a b b b a k = n a_k = n ak=n 严格递增。

在任意两个相邻的标志牌之间,汽车以恒速行驶。帖木儿有 q q q 个查询:每个查询都是一个整数 d d d ,帖木儿希望您输出汽车到达点 d d d 所需的时间,向下取整为最接近的整数

思路:

有的翻译是错的,向下取整翻译成了四舍五入了,如果有错的可以看一下。

思路还是很明显的,我们在 a a a 数组中二分找到第一个小于等于 d d d 的位置 i i i,我们再从 a i a_i ai 出发匀速走到 d d d,速度可以通过 v = a i + 1 − a i b i + 1 − b i v=\dfrac{a_{i+1}-a_{i}}{b_{i+1}-b_{i}} v=bi+1biai+1ai 计算得到。注意特判 d = a k d=a_k d=ak 的情况,因为没有后继节点,是算不出速度的。

code:

#include <iostream>
#include <cstdio>
#include <algorithm>
using namespace std;
const int maxn=1e5+5;
typedef long long ll;int T,n,k,q,d;
int a[maxn],b[maxn];int main(){cin>>T;while(T--){cin>>n>>k>>q;for(int i=1;i<=k;i++)cin>>a[i];for(int i=1;i<=k;i++)cin>>b[i];while(q--){cin>>d;if(d==n){cout<<b[k]<<" ";continue;}int i=upper_bound(a,a+k+1,d)-a-1;cout<<b[i]+1ll*(b[i+1]-b[i])*(d-a[i])/(a[i+1]-a[i])<<" ";}cout<<endl;}return 0;
}

F. Circle Perimeter

题意:

给定整数 r r r ,求与 ( 0 , 0 ) (0, 0) (0,0) 的欧氏距离大于或等于 r r r ,但严格小于 r + 1 r+1 r+1 的格点个数。大于或等于 r r r ,但严格小于 r + 1 r+1 r+1 的网格点的个数。

网格点是具有整数坐标的点。从 ( 0 , 0 ) (0, 0) (0,0) 到点 ( x , y ) (x,y) (x,y) 的欧氏距离为 x 2 + y 2 \sqrt{x^2 + y^2} x2+y2

思路:

我们可以先算出半径为 r + 1 r+1 r+1 的圆内点的个数,然后减去半径为 r r r 的圆内点的个数,答案即为所求。

听同学说是个高斯圆问题,也就是圆内整点问题。这个问题有一些数学家的猜想之类的,不过也仅限于猜想。这个题给定 ∑ r = 1 0 5 \sum r=10^5 r=105,数据范围其实并不大,所以不用数学家的奇淫寄巧,直接暴力也是可做滴。

在这里插入图片描述
我们可以先计算图中红色部分的圆内点的个数,然后乘以 4 4 4 就可以得到所有点的个数了。计算红色部分的点的个数,我们可以枚举 1 ∼ r 1\sim r 1r 的每一列(因为圆的半径只有 r r r,所以最远到第 r r r 列,外面就没点了),分别计算第 i i i 列有多少个点,累加起来即可。

这样计算出来的是半径为 r r r 的圆的点,我们再算一遍半径为 r + 1 r+1 r+1 的圆的点,后者减去前者即可。或者我们在算每一列的时候,直接算出在半径 r + 1 r+1 r+1 但不在 r r r 内的点的个数。

code:

#include <iostream>
#include <cstdio>
#include <cmath>
using namespace std;
const int maxn=1e5+5;
typedef long long ll;ll T,r;ll ge(ll x){\\最小的>=x的数if(x<0)return 0;ll t=sqrt(x);while(t*t<x)t++;return t;
}
ll lt(ll x){\\最大的<x的数ll t=sqrt(x)+1;while(t*t>=x)t--;return t;
}int main(){cin>>T;while(T--){cin>>r;ll ans=0;for(ll i=1;i<=r;i++){ans+=lt((r+1)*(r+1)-i*i)-ge(r*r-i*i)+1;} cout<<ans*4<<endl;}return 0;
}

G. XOUR

题意:

给你一个由 n n n 个非负整数组成的数组 a a a

如果 a i X O R a j < 4 a_i~\mathsf{XOR}~a_j < 4 ai XOR aj<4 ,你可以交换位置 i i i j j j 的元素,其中 X O R \mathsf{XOR} XOR 是 按位异或。

求任意交换次数所能组成的词法最小数组。

如果在 x x x y y y 相差的第一个位置上,有 x i < y i x_i < y_i xi<yi ,那么数组 x x x 在词法上比数组 y y y 小。

思路:

如果 a i ⊕ a j < 4 a_i\oplus a_j<4 aiaj<4 说明它们的二进制位在高位上是完全相同的,而最低的两位上则完全不受限制。所以我们不看低两位,如果高位相同的话,它们之间就可以相互交换位置,否则就不能。

我们把若干个高位相同的数提取出来,然后排个序,再放回空位上,对每一堆高位相同的数都做这样的操作,最后得到的就是词法最小数组。

写法应该很多,这里就给一个我自己的写法。

code:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <vector>
#include <map>
#include <queue>
#include <algorithm>
#define pii pair<int,int>
using namespace std;
const int maxn=2e5+5;int T,n,a[maxn];
map<int,priority_queue<int,vector<int>,greater<int> > > val;int main(){cin>>T;while(T--){cin>>n;for(int i=1;i<=n;i++){cin>>a[i];val[a[i]>>2].push(a[i]);}for(int i=1;i<=n;i++){auto &b=val[a[i]>>2];cout<<b.top()<<" ";b.pop();}cout<<endl;}return 0;
}

H. ±1

题意:

鲍勃有一个行数为 3 3 3 列数为 n n n 的网格,其中每一行都包含某个整数 1 ≤ i ≤ n 1 \leq i \leq n 1in a i a_i ai − a i -a_i ai 。例如, n = 4 n=4 n=4 可能包含的网格如下所示:

[ a 1 − a 2 − a 3 − a 2 − a 4 a 4 − a 1 − a 3 a 1 a 2 − a 2 a 4 ] \begin{bmatrix} a_1 & -a_2 & -a_3 & -a_2 \\ -a_4 & a_4 & -a_1 & -a_3 \\ a_1 & a_2 & -a_2 & a_4 \end{bmatrix} a1a4a1a2a4a2a3a1a2a2a3a4

爱丽丝和鲍勃玩的游戏如下:

  • 鲍勃向爱丽丝展示他的网格。
  • 爱丽丝给鲍勃一个她自己选择的数组 a 1 , a 2 , … , a n a_1, a_2, \dots, a_n a1,a2,,an ,其中的元素都是 − 1 \mathbf{-1} 1 1 \mathbf{1} 1
  • 鲍勃将这些值代入他的网格,组成一个由 − 1 -1 1 1 1 1 组成的网格。
  • 鲍勃将每一列中的元素按单调不减的顺序排列。
  • 如果中间一行的所有元素都是 1 1 1 ,则爱丽丝获胜;否则,鲍勃获胜。

例如,假设爱丽丝给鲍勃提供了上述网格的数组 [ 1 , − 1 , − 1 , 1 ] [1, -1, -1, 1] [1,1,1,1] 。那么会发生以下情况(为清晰起见,添加了颜色):

[ a 1 − a 2 − a 3 − a 2 − a 4 a 4 − a 1 − a 3 a 1 a 2 − a 2 a 4 ] → [ 1 , − 1 , − 1 , 1 ] [ 1 1 1 1 − 1 1 − 1 1 1 − 1 1 1 ] → sort each column [ − 1 − 1 − 1 1 1 1 1 1 1 1 1 1 ] \begin{bmatrix} \color{red}{a_1} & \color{green}{-a_2} & \color{blue}{-a_3} & \color{green}{-a_2} \\ -a_4 & a_4 & \color{red}{-a_1} & \color{blue}{-a_3} \\ \color{red}{a_1} & \color{green}{a_2} & \color{green}{-a_2} & a_4 \end{bmatrix} \xrightarrow{[\color{red}{1},\color{green}{-1},\color{blue}{-1},1]} \begin{bmatrix} \color{red}{1} & \color{green}{1} & \color{blue}{1} & \color{green}{1} \\ -1 & 1 & \color{red}{-1} & \color{blue}{1} \\ \color{red}{1} & \color{green}{-1} & \color{green}{1} & 1 \end{bmatrix} \xrightarrow{\text{sort each column}} \begin{bmatrix} -1 & -1 & -1 & 1 \\ \mathbf{1} & \mathbf{1} & \mathbf{1} & \mathbf{1} \\ 1 & 1 & 1 & 1 \\ \end{bmatrix}\, a1a4a1a2a4a2a3a1a2a2a3a4 [1,1,1,1] 111111111111 sort each column 111111111111
由于中间一行都是 1 1 1 ,因此爱丽丝获胜。

给定鲍勃的网格,判断爱丽丝是否可以选择数组 a a a 来赢得游戏。

思路:

年轻人的第一个 2 − s a t 2-sat 2sat,不会的就百度吧,这题很板。

有代码
详细讲解(有输出选取方案)
OI-wiki 主要是看这个 ⇒ 主要是看这个\Rightarrow 主要是看这个2-sat 各种性质的证明

上面的文档里比较重要的性质有:

  1. 2 − s a t 2-sat 2sat 构造出的原图中,关系具有对称性传递性
  2. 构造出的原图可能有强连通分量,连通分量中的点必须同时选取或者同时不选,对所有极大连通子图进行缩点,得到的新图就是一个有向无环图(DAG)。新图与原图等价,也具有对称性和传递性。
  3. 假设新图中的某个点为 S i S_i Si,对应矛盾点为 S i ′ S'_i Si对任意一对 S i , S i ′ S_i,S'_i Si,Si S i S_i Si 的后继节点与 S i ′ S'_i Si 的前代节点相互对称
  4. 若问题无解,则在原图上必然存在一对 A i , A i ′ A_i,A'_i Ai,Ai,使得 A i , A i ′ A_i,A'_i Ai,Ai 同属一个环(也就是在一个强连通分量中)。反之,如果每一对 A i , A i ′ A_i,A'_i Ai,Ai 都不属于同一个环,则问题一定有解。我们可以通过拓扑序从底向上 O ( n ) O(n) O(n) 选出一组可行解。

因为要求中间一行都是 1 1 1,所以 − 1 -1 1 的个数不能大于 1 1 1,所以如果一列三个元素的其中一个是 − 1 -1 1 的话,那么其他两个元素都必须是 1 1 1

假设这三个元素分别是 x , y , z x,y,z x,y,z 的话。如果 x x x − 1 -1 1,那么就可以推出 y , z y,z y,z 1 1 1。用 2 − s a t 2-sat 2sat 来做的话,其实也就是连边 − x → y , − x → z -x\rightarrow y,-x\rightarrow z xy,xz,同理,还有连 − y → x , − y → z , − z → x , − z → y -y\rightarrow x,-y\rightarrow z,-z\rightarrow x,-z\rightarrow y yx,yz,zx,zy

连好边之后就是很板的 2 − s a t 2-sat 2sat。我们直接检查一下每个点和它的矛盾点不在同一个强连通块内即可(也就是上面的性质 4 4 4)。

code:

#include <iostream>
#include <cstdio>
#include <vector>
using namespace std;
const int maxn=505;int T,n;
int a[maxn][3];
int id(int x){return (x<0)?-x+n:x;}vector<int> g[maxn<<1];int dfn[maxn<<1],low[maxn<<1],idx;
bool ink[maxn<<1];
int belong[maxn<<1],cnt;
//vector<vector<int> > scc;
vector<int> stk;
void tarjan(int u){dfn[u]=low[u]=++idx;stk.push_back(u);ink[u]=true;for(auto v:g[u]){if(!dfn[v]){tarjan(v);low[u]=min(low[v],low[u]);}else if(ink[v])low[u]=min(low[u],dfn[v]);}if(low[u]==dfn[u]){int nd;++cnt;do{nd=stk.back();stk.pop_back();ink[nd]=false;belong[nd]=cnt;}while(nd!=u);}
}int main(){cin>>T;while(T--){cin>>n;for(int j=0;j<3;j++)for(int i=1;i<=n;i++)cin>>a[i][j];idx=cnt=0;for(int i=1;i<=2*n;i++){g[i].clear();dfn[i]=low[i]=belong[i]=0;}for(int i=1,x,y,z;i<=n;i++){x=a[i][0];y=a[i][1];z=a[i][2];
//			cout<<x<<" "<<y<<" "<<z<<endl;g[id(-x)].push_back(id(y));g[id(-x)].push_back(id(z));g[id(-y)].push_back(id(x));g[id(-y)].push_back(id(z));g[id(-z)].push_back(id(x));g[id(-z)].push_back(id(y));}for(int u=1;u<=2*n;u++)if(!dfn[u])tarjan(u);bool flag=true;for(int u=1;u<=n;u++){if(belong[id(u)]==belong[id(-u)])flag=false;}puts((flag)?"YES":"NO");}return 0;
} 

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

棒材直线度测量仪 专为圆形产品研发设计 在线无损检测

棒材直线度测量仪采用了先进的技术&#xff0c;能够实现在线无损检测&#xff0c;为生产过程提供了极大的便利。专为圆形产品设计&#xff0c;它能够精确测量棒材的米直线度及外径、椭圆度尺寸&#xff0c;为质量控制提供可靠的数据支持。 在线直线度测量仪不仅具有出色的性能…

Stm32串口搭配DMA实现自定义printf、scanf

前言:本文仅供学习参考使用&#xff0c;主要目的是让大家快速使用串口调试&#xff0c;文章所提及的GCC适用于Clion&#xff0c;Vscode等第三方编辑器的用户。作者有时间会继续更新^_^ 一、GCC环境 1、标准库 (1)、使用方法 在主函数while(1)初始化中&#xff0c;添加Seria…

嵌入式学习-M4的基本定时器

基本介绍 框图分析 时钟选择 计数器结构 开启重装载值寄存器的影子寄存器的工作时序图 未开启重装载值寄存器的影子寄存器的工作时序图 更新事件以及中断 相关寄存器 相关库函数

全球排名第一的质量管理(QMS)系统介绍,100%免费开源

什么是Odoo全程质量管理&#xff1f; 开源智造Odoo免费开源质量管理系统将政策、标准和实践规范化并自动化&#xff0c;以最小的开销提供最高质量的产品。快速轻松地选择要执行的测试&#xff0c;设置参数以评估结果&#xff0c;并定义接收时和生产过程中的测试策略。Odoo会自动…

OpenNJet产品体验:探索无限可能

文章目录 前言一、OpenNJet是什么&#xff1f;二、OpenNJet特性和优点三、OpenNJet功能规划四、OpenNJet快速上手五、OpenNJet的使用总结 前言 现代社会网络高速发展&#xff0c;同时也迎来了互联网发展的高峰&#xff0c;OpenNJet作为一个基于NGINX的面向互联网和云原生应用提…

网络安全等级保护测评指标解读(2.0)

网络安全等级保护2.0测评指标解读&#xff1a; 1、物理和环境安全 2、网络和通信安全 3、设备和计算安全 4、应用和数据安全 5、安全策略和管理制度 6、安全管理机构和人员 7、安全建设管理 8、安全运维管理 软件资料清单列表部分文档&#xff1a; 工作安排任务书&#xff0c;…

HCIA和HCIP区别大吗?小白请看这

华为认证以其专业性和实用性受到了业界的广泛认可。 HCIA、HCIP、HCIP这三个级别&#xff0c;你会选哪个&#xff1f;IE含金量不用多说&#xff0c;IA还是IP&#xff0c;你会纠结吗。 但面对这两个级别的认证&#xff0c;初学者或者“小白”们可能会感到困惑&#xff1a;两者…

python “名称空间和作用域” 以及 “模块的导入和使用”

七、名称空间和作用域 可以简单理解为存放变量名和变量值之间绑定关系的地方。 1、名称空间 在 Python 中有各种各样的名称空间&#xff1a; 全局名称空间&#xff1a;每个程序的主要部分定义了全局的变量名和变量值的对应关系&#xff0c;这样就叫做全局名称空间 局部名称…

Ajax额

原生Ajax xml 已被json取代 http 请求方法urlhttp版本号 network 谷歌浏览器查看请求报文和响应报文 F12 network header里面有 请求头 响应头 点击view source 可以查看请求响应行 请求体在请求行头下面 get请求有url参数&#xff0c;请求体变为query String…

基于springboot实现的家具销售电商平台

开发语言&#xff1a;Java 框架&#xff1a;springboot JDK版本&#xff1a;JDK1.8 服务器&#xff1a;tomcat7 数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09; 数据库工具&#xff1a;Navicat11 开发软件&#xff1a;eclipse/myeclipse/idea Maven包&…

超分辨率技术

何为超分辨率&#xff1f; 超分辨率技术&#xff08;Super Resolution&#xff09;&#xff1a;通过硬件或软件的方法提高图像或视频帧的分辨率&#xff0c; 通过一系列低分辨率图像获取到高分辨率图像的过程。 功能介绍&#xff1a;提升图像或视频分辨率最高至8K&#xff0c;…

C#窗体程序设计笔记:如何调出控件工具箱,并设置控件的属性

文章目录 调出控件工具箱设置控件属性 调出控件工具箱 使用Visual Studio打开C#解决方案后&#xff0c;初始界面如下图所示&#xff1a; 接着&#xff0c;在上方的菜单栏依次选择“视图”“工具箱”&#xff0c;即可打开工具箱&#xff0c;如下图所示&#xff1a; 设置控件属…