洛谷P3516 PRZ-Shift [POI2011] 构造

时间:2022-03-10 09:54:00

正解:构造

解题报告:

传送门!

umm这题就是很思维的?就是想到了就A了想不到就做不出来,然而我也只能是做到理解不知道怎么想出来,,,感觉构造题什么的就很真诚,一点套路也没有,所以像我这种没有脑子只会投机取巧找规律的弱鸡就完全想不到QAQ撑死也不够是能理解解法而已QAQ希望多做一点这类的题目能有帮助趴QAQ

首先嗦嗦这题最基本的变化,就是,假如我现在已经确定了一段[1,i]了,现在要把第i+1放到第i个后面去,假如是酱婶的:balabala 1 2 ... i balabala i+1 balabala(事实上最前面那个balabala是不可能存在的存在也很容易消掉的,但是不要在意这种细节反正没有影响×只是代码有点儿区别注意下就是了qwq

显然的是当[1,i]后面乜有数了并且第i+1个在第一个的时候就能用i个操作a于是就能得到[1,i+1]balabala了,然后一直这么做做做做下去就欧克了

那怎么实现呢,首先用一堆a把第i+1个放到第一个,这儿不难?然后考虑怎么做到把[1,i]之后那些杂七杂八的数都移到前面去并且保持i+1在第一个?其实比较容易想到?就用俩A然后i+1就到了第三位这时候再用个B,i+1就又到第一个了

然后注意一下的是如果到最后i+1在第一个,就直接做;如果在第三个,用一次B然后直接做;如果在第二个呢?当然也不难咯,用两次B就可以了

另外一个需要注意的地方是当解决得差不多了只剩最后两个的时候,如果出现了n,n-1,1,2,...,n-2的情况,按照我们之前的思路,就是用两次B然后就能让n-1到第一位然后就这么顺利地做下去辽

但是!显然这个情况下就不欧克了鸭QAQ因为n被夹到[1,n-2]这个区间里去了就废了

所以考虑换一种变化方式

就是,直接假装n-1在第一个时候的样子移动[1,n-2],不过是当第n个在第三个的时候使用B

这样最后的效果应该就是变成酱婶的:n,1,2,...,n-1

大概能get到?

然后再走n-1次A就欧克了!

然后如果在刚刚说的情况中,n是奇数,辣就是说n-2是奇数,就无法每次都移动俩,所以就无解输出NIE

over!

umm然后放个代码QwQ

#include<bits/stdc++.h>
using namespace std;
#define rp(i,x,y) for(register int i=x;i<=y;++i) const int N=;
int n,a[N],b[N],op[N*N],len[N*N],ans,flag; inline int read()
{
char ch=getchar();int x=;bool y=;
while(ch!='-' && (ch>'' || ch<''))ch=getchar();
if(ch=='-')ch=getchar(),y=;
while(ch>='' && ch<='')x=(x<<)+(x<<)+(ch^''),ch=getchar();
return y?x:-x;
}
void mva(int l){rp(i,,n)b[(i+l)%n]=a[i];b[n]=b[];rp(i,,n)a[i]=b[i];}
void mvb(int &x,int &y,int &z){int tmp=z;z=y,y=x,x=tmp;}
void fd(int i,int &x){rp(j,,n)if(a[j]==i){x=j;return;}}
void final(int x)
{
int w=n+,mva=(n-x)&;
for(int i=x+mva;a[i+]!=;i--)b[--w]=a[i];w=;rp(i,x++mva,n)b[++w]=a[i];
for(int i=;a[i]!=;i++)b[++w]=a[i];rp(i,,n)a[i]=b[i];
if(!mva)return;
rp(i,,n-)b[i+]=a[i];
b[]=a[];b[]=a[],b[]=a[n];
rp(i,,n)a[i]=b[i];
}
void ad(int l,int k)
{
if(op[ans]==k)len[ans]+=l;else len[++ans]=l,op[ans]=k;
if(op[ans]==)len[ans]%=n;else len[ans]%=;
if(!len[ans])--ans;
}
void printans()
{
if(flag){printf("NIE\n");return;}
printf("%d\n",ans);
rp(i,,ans)
{
printf("%d%c",len[i],op[i]+'a'-);
if(i!=ans)printf(" ");
}
}
void dolast()
{
int x;fd(,x);int tt=(n-x+)%n;
if(tt)mva(tt),ad(tt,);
if(a[n-]==n-)return;
if(n&){flag=;return;}
mva(),ad(,);
while(a[n]!=n-){ad(,);mva();ad(,);mvb(a[],a[],a[]);}
ad(n-,);mva(n-);
}
int main()
{
n=read();ans=;flag=;
rp(i,,n)a[i]=read();
rp(i,,n-)
{
int x;fd(i,x);if(x==i)continue;int tt=n-x+;
mva(tt);ad(tt,);fd(i-,x);
rp(k,,(n-x)>>)ad(,),ad(,);
if((n-x)&)ad(,),ad(,);
ad(i-,);final(x);mva(i-);
}
dolast();printans();
}

好难打啊,,,死亡QAQ