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->1
		for(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;
} 
  • 26
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值