[Luogu5241]序列(DP)

固定一种构造方法,使它能够构造出所有可能的序列。

对于一个要构造的序列,把所有点排成一串,若a[i]=a[i-1],那么从1所在弱连通块往连通块后一个点连,若所有点都在一个连通块里了,就在1所在强连通块中随便连。若a[i]<a[i-1],那么连一条从后往前的边让若干点与1所在强连通快合并。显然这样可以得到所有可能的序列。

于是前一部分“连弱连通块”用f[i][k][x]表示前i个点在一个弱连通块中,到目前为止序列共减小了k次,1所在连通块大小为x,的方案数(此时边数为i-1+k)。

后一部分用g[i][x]表示图中共i条边,1所在连通块大小为x的方案数。

两个状态数都是$O(n^3)$的且可以用前缀和$O(1)$转移,g[][]的初值由f[][][]得到。

 1 #include<cstdio>
 2 #include<algorithm>
 3 #define rep(i,l,r) for (int i=(l); i<=(r); i++)
 4 using namespace std;
 5 
 6 const int N=410,mod=1e9+7;
 7 int n,f[N][N][N],g[N*N][N],smf[N][N][N],smg[N*N][N],res[N*N];
 8 
 9 int inc(int x,int y){ x+=y; return x>=mod ? x-mod : x; }
10 
11 int main(){
12     freopen("c.in","r",stdin);
13     freopen("c.out","w",stdout);
14     scanf("%d",&n); f[1][0][1]=smf[1][0][1]=1;
15     rep(i,2,n) rep(k,0,i-1) rep(x,1,i){
16         f[i][k][x]=f[i-1][k][x];
17         if (k && i<n) f[i][k][x]=inc(f[i-1][k][x],smf[i][k-1][x-1]);
18         smf[i][k][x]=inc(smf[i][k][x-1],f[i][k][x]);
19         if (i<n) res[i-1+k]=inc(res[i-1+k],f[i][k][x]);
20         if (i==n) g[i-1+k][x]=inc(g[i-1+k][x],f[i][k][x]);
21     }
22     rep(i,1,n*(n-1)) rep(x,1,n) if (i<=n*(n-1)/2+x*(x-1)/2){
23         g[i][x]=inc(inc(g[i][x],g[i-1][x]),smg[i-1][x-1]);
24         smg[i][x]=inc(smg[i][x-1],g[i][x]);
25     }
26     rep(i,1,n*(n-1)){
27         int ans=0;
28         rep(x,1,n) ans=inc(ans,g[i][x]);
29         printf("%d ",inc(ans,res[i]));
30     }
31     return 0;
32 }

 

posted @ 2019-03-05 14:14  HocRiser  阅读(251)  评论(0编辑  收藏  举报