@ysner
2018-10-16T09:16:56.000000Z
字数 1661
阅读 2512
DP 排列
给定一个的排列。一个的排列被认为是优美
的排列,当且仅当满足下列条件:
对排列进行次交换。
最后能使得排列
问有多少个优美的排列,答案对取模。
首先显而易见的是这个排列有对逆序对。
能影响每个位置的值的交换,只有两个:
然后想想把调到其在中的位置需要满足哪些条件。
若,说明要向左调,因此其前面一个的交换应比其自己的交换靠前。
而在中,它们自己的交换应比其前面一个的交换靠前,否则无法保证到达。
这样就能得出这个交换的相对顺序。
设为统计到第个位置的交换,该交换在排列中排第个。
然后再枚举上一个位置交换排第几个,依限制转移即可,复杂度。
发现可以转移的部分不是前缀就是后缀,可以前缀和优化,复杂度降到。
嗯,还有个只能让蒟蒻挂掉的细节:
的转移是合法的。
因为从前往后转移的过程实际上是插入数的过程,所以如果第个位置的交换在第个,后面所有交换的位置都会后移一位。
#include<iostream>#include<cmath>#include<cstdio>#include<cstdlib>#include<cstring>#include<algorithm>#define ll long long#define re register#define il inline#define fp(i,a,b) for(re int i=a;i<=b;++i)#define fq(i,a,b) for(re int i=a;i>=b;--i)using namespace std;const int N=5e3+100,mod=1e9+7;int n,pos[N],ans,op[N],tag=1,f[N][N],g[N][N];bool vis[N];il ll gi(){re ll x=0,t=1;re char ch=getchar();while(ch!=-'-'&&(ch<'0'||ch>'9')) ch=getchar();if(ch=='-') t=-1,ch=getchar();while(ch>='0'&&ch<='9') x=x*10+ch-48,ch=getchar();return x*t;}//op前面操作先于后面操作1 or 前面操作后于后面操作2il void Pre(){fp(i,1,n){if(pos[i]==i) tag=0;if(pos[i]<i){if(pos[i]!=1) op[i]|=1;fp(j,pos[i]+1,i-1) op[j]|=2;}if(pos[i]>i){if(pos[i]!=n) op[i]|=2;fp(j,i+1,pos[i]-1) op[j]|=1;}}fp(i,1,n-1) if(op[i]==3) tag=0;}int main(){n=gi();fp(i,1,n) pos[gi()+1]=i;Pre();if(!tag) return puts("0"),0;f[1][1]=g[1][1]=1;fp(i,2,n-1){fp(j,1,i){if(op[i]==0) (f[i][j]+=g[i-1][i-1])%=mod;if(op[i]==1) (f[i][j]+=g[i-1][j-1])%=mod;if(op[i]==2) (f[i][j]+=g[i-1][i-1]-g[i-1][j-1]+mod)%=mod;}fp(j,1,i) g[i][j]=(g[i][j-1]+f[i][j])%mod;}printf("%d\n",g[n-1][n-1]);return 0;}
