D. Carousel

题意

给你一个长度为 n ( n > 3 ) n(n>3) n(n>3)的数组,首尾相连形成一个环。对每个位置进行涂色,要求相邻但不相同数字的填涂颜色不同,相邻且相同的位置颜色可以相同也可以不相同。要求所用颜色的最大数字最小,颜色由数字来表示。

思路

分类讨论 构造

  • 全部为同一种颜色,则全为1即可。
  • 至少有一个位置与其相邻的数字相同,则我们可以从这个相同的位置剖开这个环,形成一条线,再从头到尾1,2交替的填入即可。
  • n为偶数,交替填入1,2即可,首尾必不相同。
  • 全部不相同且为奇数,第一个填入3,后面交替填入1,2即可。

这道题需要仔细严谨的分情况去考虑,关键点在于抓住题干中的两相邻数字相同时可相同也可不同,从而想到两相邻数字相同时,可以从中剖开交替填入1,2。

解锁绑定cin/scanf cout/printf之后要注意不能用puts putchar getchar之类的
don’t mix cin/scanf, cout/printf!

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int,int>P;
const double eps = 1e-8;
const int NINF = 0xc0c0c0c0;
const int INF  = 0x3f3f3f3f;
const ll  mod  = 1e9 + 7;
const ll  maxn = 2e5 + 5;

int n,a[maxn];

bool jud1(){
	bool flag=0;
	for(int i=1;i<=n;i++) if(a[i]!=a[i%n+1]){flag=1;break;}
	if(flag) return 0;
	cout<<"1\n";
	for(int i=1;i<=n;i++) cout<<1<<" ";
	puts("");
    return 1;
}

bool jud2(){
	bool flag=0;int pos=0;
	for(int i=1;i<=n;i++) if(a[i]==a[i%n+1]){pos=i;flag=1;break;}
	if(!flag&&(n%2)) return 0;
	cout<<2<<"\n";
	for(int i=1;i<=n;i++){//这下面的代码很丑 某大佬WZYYN用的是 printf("%d ",1+(i-p+n)%n%2);一句话就搞定了从中剖开交替输出的含义。
		if(i<=pos){
			if((pos-i)&1){
				cout<<"1 ";
			}else cout<<"2 ";
		}else{
			if(pos&1){
				if((n-i)&1) cout<<"2 ";
				else cout<<"1 ";
			}else{
				if((n-i)&1) cout<<"1 ";
				else cout<<"2 ";
			}
		}
	}
	puts("");
    return 1;
}

void jud3(){
	cout<<"3\n3";
	for(int i=2;i<=n;i++) cout<<" "<<1+(i&1);
	puts("");
}

void solve(){
	cin>>n;
	for(int i=1;i<=n;i++) cin>>a[i];
	if(jud1()) return;
    if(jud2()) return;
    jud3();
}

int main(){
	int T;cin>>T;
	while(T--) solve();
	return 0;
}