[luogu7418]Counting Graphs P

参考[luogu7417],同样求出最短路,得到二元组$(x,y)$并排序,记$tot_{(x,y)}$为$(x,y)$的数量

其中所给的两个条件,即分别要求:

1.$(x,y)$只能和$(x\pm 1,y\pm 1)$连边

2.每一个$(x,y)$都向$(x-1,y\pm 1)$中的一个连边、$(x\pm 1,y-1)$中的一个连边

(另外,注意在$x+1=y$时$(x+1,y-1)$也即为$(x,y)$自身)

从前往后依次dp,假设考虑到二元组为$(x,y)$,将之前二元组分为两类——

(1)对于其中不为$(x,y)$的二元组之后已经不存在能通过连边满足第2个条件的点了,那么就需要保证第2个条件都已经满足,即状态中不需要再记录

(2)对于二元组$(x,y)$,之后也不存在$(x-1,y\pm 1)$的点了,因此$x$的一维必须要已经满足,并记录$y$的一维不满足的数个数即可

由此,即得到一个二维的状态,将其记为$f_{(x,y),i}$(其中$i$的定义参考(2)中)

关于转移,有以下四类本质不同的边:

(1)$(x,y)$与$(x-1,y+1)$中$y$的一维已经满足点的的边

(2)$(x,y)$与$(x-1,y+1)$中$y$的一维仍未满足的点的边

(3)$(x,y)$与$(x-1,y-1)$的边

(4)若$x+1=y$,$(x,y)$和$(x,y)$之间的边

先转移前两类边(避免出现三维状态),枚举后者的点数,转移即
$$
f_{(x,y),i}={tot_{(x,y)}\choose i}\sum_{j=0}^{tot_{(x-1,y+1)}}G(tot_{(x-1,y+1)},j,tot_{(x,y)}-i)f_{(x-1,y+1),j}
$$
(注意此时的$i$的定义并不为最终的定义,仅是一个中间过程)

其中$G(x,z,y)$表示在左边$x$个点和右边$y$个点连边的$2^{xy}$张图中,满足左边特定的$z$个点(如前$z$个点)和右边的$y$​个点度数均非0的方案数,对左边容斥有
$$
G(x,z,y)=\sum_{i=0}^{z}(-1)^{i}{z\choose i}(2^{x-i}-1)^{y}
$$
再转移第3类边,考虑转移的状态$f_{(x,y),j}$,这$j$个点必须都选择(否则$x$这一维不满足),因此转移即
$$
f_{(x,y),i}=\sum_{j=0}^{tot_{(x,y)}}{tot_{(x,y)}-j\choose i}(2^{tot_{(x-1,y-1)}}-1)^{tot_{(x,y)}-i}f_{(x,y),j}
$$
(这里是类似于01背包的,即后者$f_{(x,y),j}$应该是未转移第3类边的结果)

最后转移第4类边,其实这只是一个特例,更完整的情况应该是对于$(x+1,y-1)$不再出现的点(即其之后不再参与转移,之后也没有点可以帮助其满足第2个条件)将其的贡献乘到答案上

具体的,这类状态又分为两类:

1.$x+1\ne y$,这类状态不能通过自环使其满足条件,因此贡献即$f_{(x,y),0}$

2.$x+1=y$,这还可以通过自环使其满足条件

具体的,贡献即$\sum_{i=0}^{tot_{(x,y)}}G(tot_{(x,y)},i)f_{(x,y),i}$,其中$G(x,y)$表示在$x$个点中任意连边的$2^{\frac{x(x+1)}{2}}$张图中(允许自环),满足特定的$y$个点度数均非0的方案数,对这$y$​个点容斥有
$$
G(x,y)=\sum_{i=0}^{y}(-1)^{i}{y\choose i}2^{\frac{(x-i)(x-i+1)}{2}}
$$
不难发现$f$的状态数实际仅为$o(n)$,转移复杂度为$o(n^{2})$,总复杂度即$o(n^{3})$,可以通过

  1 #include<bits/stdc++.h>
  2 using namespace std;
  3 #define N 105
  4 #define M 10005
  5 #define mod 1000000007
  6 #define oo 0x3f3f3f3f
  7 #define ll long long
  8 struct Edge{
  9     int nex,to;
 10 }edge[M*3];
 11 queue<int>q;
 12 int t,n,m,E,x,y,ans,mi[M],Mi[N][N],C[N][N],head[N<<1],d[N<<1],tot[N][N],g[N],f[N];
 13 void add(int x,int y){
 14     edge[E].nex=head[x];
 15     edge[E].to=y;
 16     head[x]=E++;
 17 }
 18 int G(int x,int y){
 19     int ans=0;
 20     for(int i=0;i<=y;i++){
 21         int s=(ll)C[y][i]*mi[(x-i)*(x-i+1)/2]%mod;
 22         if (i&1)ans=(ans-s+mod)%mod;
 23         else ans=(ans+s)%mod;
 24     }
 25     return ans;
 26 }
 27 int G(int x,int z,int y){
 28     int ans=0;
 29     for(int i=0;i<=z;i++){
 30         int s=(ll)C[z][i]*Mi[x-i][y]%mod;
 31         if (i&1)ans=(ans-s+mod)%mod;
 32         else ans=(ans+s)%mod;
 33     }
 34     return ans;
 35 }
 36 int main(){
 37     mi[0]=1;
 38     for(int i=1;i<M;i++)mi[i]=2*mi[i-1]%mod;
 39     for(int i=0;i<N;i++){
 40         Mi[i][0]=1;
 41         for(int j=1;j<N;j++)Mi[i][j]=(ll)Mi[i][j-1]*(mi[i]-1)%mod;
 42     }
 43     for(int i=0;i<N;i++){
 44         C[i][0]=C[i][i]=1;
 45         for(int j=1;j<i;j++)C[i][j]=(C[i-1][j-1]+C[i-1][j])%mod;
 46     }
 47     scanf("%d",&t);
 48     while (t--){
 49         scanf("%d%d",&n,&m);
 50         E=0,ans=1;
 51         memset(head,-1,sizeof(head));
 52         memset(d,oo,sizeof(d));
 53         memset(tot,0,sizeof(tot));
 54         for(int i=1;i<=m;i++){
 55             scanf("%d%d",&x,&y);
 56             add(x,y+n),add(y+n,x);
 57             add(x+n,y),add(y,x+n);
 58         }
 59         d[1]=0;
 60         q.push(1);
 61         while (!q.empty()){
 62             int k=q.front();
 63             q.pop();
 64             for(int i=head[k];i!=-1;i=edge[i].nex)
 65                 if (d[edge[i].to]==oo){
 66                     d[edge[i].to]=d[k]+1;
 67                     q.push(edge[i].to);
 68                 }
 69         }
 70         if (d[n+1]==oo){
 71             for(int i=1;i<=n;i++)tot[0][min(d[i],d[i+n])]++;
 72             for(int i=1;i<=n;i++)ans=(ll)ans*Mi[tot[0][i-1]][tot[0][i]]%mod;
 73             printf("%d\n",ans);
 74             continue;
 75         }
 76         for(int i=1;i<=n;i++)tot[d[i]+d[i+n]-d[n+1]>>1][min(d[i],d[i+n])]++;
 77         for(int x=0;x<=n;x++){
 78             memset(f,0,sizeof(f));
 79             f[0]=1;
 80             for(int y=x;y<=n;y++){
 81                 if (!tot[x][y]){
 82                     if (y-x<=(d[n+1]>>1))ans=(ll)ans*f[0]%mod;
 83                     else{
 84                         int s=0;
 85                         for(int i=0;i<=tot[x][y-1];i++)s=(s+(ll)G(tot[x][y-1],i)*f[i])%mod;
 86                         ans=(ll)ans*s%mod;
 87                     }
 88                     memset(f,0,sizeof(f));
 89                     f[0]=1;
 90                     continue;
 91                 }
 92                 if ((!x)&&(!y)){
 93                     f[0]=0,f[1]=1;
 94                     continue;
 95                 }
 96                 memcpy(g,f,sizeof(g));
 97                 memset(f,0,sizeof(f));
 98                 for(int i=0;i<=tot[x][y];i++){
 99                     int s=0;
100                     for(int j=0;j<=tot[x][y-1];j++)s=(s+(ll)G(tot[x][y-1],j,tot[x][y]-i)*g[j])%mod;
101                     f[i]=(f[i]+(ll)C[tot[x][y]][i]*s)%mod;
102                 }
103                 memcpy(g,f,sizeof(g));
104                 memset(f,0,sizeof(f));
105                 if (!x){
106                     f[tot[x][y]]=g[0];
107                     continue;
108                 }
109                 for(int i=0;i<=tot[x][y];i++)
110                     for(int j=0;j<=tot[x][y];j++)f[i]=(f[i]+(ll)C[tot[x][y]-j][i]*Mi[tot[x-1][y-1]][tot[x][y]-i]%mod*g[j])%mod;
111             }
112         }
113         printf("%d\n",ans);
114     }
115     return 0;
116 } 
View Code

 

posted @ 2021-08-28 06:38  PYWBKTDA  阅读(75)  评论(0编辑  收藏  举报