cf题面数组
给个n,再给两个长度为2的字符串,要求构造一个长度为\(3n\)的字符串,a、b、c三个字母各n个,且构造出的字符串子串中不能出现给定的两个字符串。若是不存在这样的字符串,就输出NO
函数
首先生成a、b、c的6个全排列,而后,每种全排列以两种方式扩大n倍——举个例子,n为3时,对于acb
,能够扩展成这两种:acbacbacb
、aaacccbbb
。而后依次检查这12种字符串是否有知足要求的,有就输出,没有就表明没有了(能够考虑字符串排列的内部以及扩展后字符串之间的边界)spa
(虚拟赛时想到第一种扩展方式,把本身卡了之后,想到第二种,又把本身卡了,愣是想不到两种都试试……).net
#include<cstdio> #include<stdlib.h> #include<algorithm> const int MAXN=3e5+5; int n; char mo[6][4]={"abc","acb","bac","bca","cab","cba"}; char input[2][3]; char s[MAXN]; void check() { for(int i=2;i<=n*3;i++) { if(s[i]==input[0][1]&&s[i-1]==input[0][0]||s[i]==input[1][1]&&s[i-1]==input[1][0]) return; } printf("YES\n%s",s+1); exit(0); } int main() { scanf("%d%s%s",&n,input[0],input[1]); for(int m=0;m<6;m++) { for(int i=1;i<=n;i++)//而后一个奇怪的地方,我以前提交的时候input数组开到[2][2],输入时显然溢出了,可是输入那里不报错,反而这个循环不会进入,就是i这个循环。m那个循环进来了,而后直接进入check函数。为何嘞……留坑 { s[(i-1)*3+1]=mo[m][0]; s[(i-1)*3+2]=mo[m][1]; s[(i-1)*3+3]=mo[m][2]; } check(); for(int i=1;i<=n;i++) { s[i]=mo[m][0]; s[i+n]=mo[m][1]; s[i+n+n]=mo[m][2]; } check(); } puts("NO"); return 0; }
本博客第一篇被打上构造
标签的博文(真该打个标签叫智商
)code