传送门:洛谷
设 f ( i ) f(i) f(i)表示 i i i个点的无向联通图的个数.设 g ( i ) g(i) g(i)为 i i i个点的无向图的个数.
考虑枚举一个点的联通分量包含点的个数,不妨标记为1号点.那么有: g ( n ) = ∑ i = 1 n f ( i ) ∗ C n − 1 i − 1 ∗ g ( n − i ) g(n)=\sum_{i=1}^nf(i)*C_{n-1}^{i-1}*g(n-i) g(n)=i=1∑nf(i)∗Cn−1i−1∗g(n−i)
上述式子的含义就是枚举一号点的联通分量的个数,那么对于枚举的个数 i i i,我们都可以选出除了 1 1 1号点的 n − 1 n-1 n−1个点中的 i − 1 i-1 i−1个点作为我们的与 1 1 1联通的点.该部分的方案数就是 i i i个点的无向联通图的个数.那么对于剩下的 n − i n-i n−i个来说,我们是无所谓的.所以该部分的方案数就是无向图的个数.
此时会发现,我们需要 g ( 0 ) = 1 g(0)=1 g(0)=1,但是我们光光由上述的式子无法确定 g ( 0 ) = 1 g(0)=1 g(0)=1,所以我们其实是需要考虑补充定义的,这部分全网几乎没有人提及.所以我们考虑补充定义: g ( n ) = [ n = 0 ] + ∑ i = 1 n f ( i ) ∗ C n − 1 i − 1 ∗ g ( n − i ) g(n)=[n=0]+\sum_{i=1}^nf(i)*C_{n-1}^{i-1}*g(n-i) g(n)=[n=0]+i=1∑nf(i)∗Cn−1i−1∗g(n−i).
接下来考虑 n > 0 n>0 n>0的情况(因为我们 g ( 0 ) = 1 g(0)=1 g(0)=1是规定好的),我们化解一下我们的式子:
g ( n ) = ∑ i = 1 n f ( i ) ∗ g ( n − i ) ∗ ( n − 1 ) ! ( i − 1 ) ! ∗ ( n − i ) ! g(n)=\sum_{i=1}^nf(i)*g(n-i)*\frac{(n-1)!}{(i-1)!*(n-i)!} g(n)=i=1∑nf(i)∗g(n−i)∗(i−1)!∗(n−i)!(n−1)! g ( n ) ( n − 1 ) ! = ∑ i = 1 n f ( i ) ( i − 1 ) ! ∗ g ( n − i ) ( n − i ) ! \frac{g(n)}{(n-1)!}=\sum_{i=1}^n\frac{f(i)}{(i-1)!}*\frac{g(n-i)}{(n-i)!} (n−1)!g(n)=i=1∑n(i−1)!f(i)∗(n−i)!g(n−i)
考虑设 F = ∑ i = 1 n f ( i ) ( i − 1 ) ! F=\sum_{i=1}^n\frac{f(i)}{(i-1)!} F=∑i=1n(i−1)!f(i), G = ∑ i = 0 n g ( i ) i ! , H = ∑ i = 1 n i ( i − 1 ) ! G=\sum_{i=0}^{n}\frac{g(i)}{i!},H=\sum_{i=1}^n\frac{i}{(i-1)!} G=∑i=0ni!g(i),H=∑i=1n(i−1)!i
发现是一个卷积的形式. H = F ∗ G H=F*G H=F∗G
PS:此时可能会有人会有疑问了,我们此时为什么不需要加上 1 1 1,这是由于我们左边的0次项系数为0,右边F的0此项系数也为0,G的0次项系数为1,我们此时不加上1才是满足等式成立的
然后此时我们的 F = H ∗ G − 1 F=H*G^{-1} F=H∗G−1,使用多项式求逆和多项式乘法即可
下面是具体的代码部分:
#include
using namespace std;
typedef long long ll;
#define root 1,n,1
#define ls rt<<1
#define rs rt<<1|1
#define lson l,mid,rt<<1
#define rson mid+1,r,rt<<1|1
inline ll read() {
ll x=0,w=1;char ch=getchar();
for(;ch>'9'||ch<'0';ch=getchar()) if(ch=='-') w=-1;
for(;ch>='0'&&ch<='9';ch=getchar()) x=x*10+ch-'0';
return x*w;
}
inline void print(__int128 x){
if(x<0) {putchar('-');x=-x;}
if(x>9) print(x/10);
putchar(x%10+'0');
}
#define maxn 1000000
#define int long long
const int mod=1004535809;
const double eps=1e-8;
#define int_INF 0x3f3f3f3f
#define ll_INF 0x3f3f3f3f3f3f3f3f
int qpow(int a,int b) {
int ans=1;
while(b) {
if(b&1) ans=ans*a%mod;
b>>=1;
a=a*a%mod;
}
return ans;
}
int rev[maxn];
void NTT(int *a,int n,int inv) {
for(int i=0;i<=n;i++) if(i<rev[i]) swap(a[i],a[rev[i]]);
for(int len=1;len<=(n>>1);len<<=1) {
int gn=qpow(inv==1?3:qpow(3,mod-2),(mod-1)/(len<<1));
for(int i=0;i<=n;i+=(len<<1)) {
int g0=1;
for(int j=0;j<=len-1;j++) {
int x=a[i+j],y=a[i+j+len]*g0;
a[i+j]=(x+y)%mod;a[i+j+len]=((x-y)%mod+mod)%mod;
g0=g0*gn%mod;
}
}
}
}
int INV_temp[maxn];
void INV(int *a,int *b,int deg) {
if(deg==1) {b[0]=qpow(a[0],mod-2);return ;}
INV(a,b,(deg+1)>>1);
for(int i=0;i<deg;i++) INV_temp[i]=a[i];
int limit=1,len=0;
while(limit<=deg-1+deg-1) limit<<=1,len++;
for(int i=0;i<=limit;i++) rev[i]=(rev[i>>1]>>1)|((i&1)<<(len-1));
NTT(INV_temp,limit,1);NTT(b,limit,1);
for(int i=0;i<=limit;i++) {
b[i]=((2*b[i]-INV_temp[i]*b[i]%mod*b[i]%mod)%mod+mod)%mod;
}
NTT(b,limit,-1);
int inv=qpow(limit,mod-2);
for(int i=0;i<=limit;i++) b[i]=b[i]*inv%mod;
for(int i=deg;i<=limit;i++) b[i]=0;
for(int i=0;i<=limit;i++) INV_temp[i]=0;
}
int fac[maxn],in_fac[maxn];
void init(int limit) {
fac[0]=in_fac[0]=1;
for(int i=1;i<=limit;i++) {
fac[i]=fac[i-1]*i%mod;
in_fac[i]=in_fac[i-1]*qpow(i,mod-2)%mod;
}
}
int g[maxn],h[maxn],inv_g[maxn],f[maxn];
signed main() {
int n=read();
init(n);
for(int i=0;i<=n;i++) {
g[i]=qpow(2,i*(i-1)/2)*in_fac[i]%mod;
}
for(int i=1;i<=n;i++) {
h[i]=qpow(2,i*(i-1)/2)*in_fac[i-1]%mod;
}
INV(g,inv_g,n+1);
int limit=1,len=0;
while(limit<=n+n) limit<<=1,len++;
for(int i=0;i<=limit;i++) rev[i]=(rev[i>>1]>>1)|((i&1)<<(len-1));
NTT(h,limit,1);NTT(inv_g,limit,1);
for(int i=0;i<=limit;i++) f[i]=h[i]*inv_g[i]%mod;
NTT(f,limit,-1);
int inv=qpow(limit,mod-2);
for(int i=0;i<=limit;i++) f[i]=f[i]*inv%mod;
cout<<fac[n-1]*f[n]%mod<<endl;
return 0;
}