bzoj 3456 城市规划 —— 分治FFT / 多项式求逆 / 指数型生成函数(多项式求ln)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3456
首先考虑DP做法,正难则反,考虑所有情况减去不连通的情况;
而不连通的情况就是那个经典做法:选定一个划分点,枚举包含它的连通块,连通块以外的部分随便连(但不和连通块连通),合起来就是不连通的方案数;
设 \( f[i] \) 表示一共 \( i \) 个点时的连通方案数,\( g[i] \) 表示 \( i \) 个点随便连的方案数,即 \( g[i] = 2^{C_{i}^{2}} \),则:
\( f[i] = 2^{C_{i}^{2}} - \sum\limits_{j=1}^{i-1} C_{i-1}^{j-1} * f_{j} * g_{i-j} \)
只要令 \( g[0] = 0 \),\( j \) 就可以枚举到 \( i \);
把 \( C_{i-1}^{j-1} \) 拆开,对应分配到各处,得到:
\( f[i] = 2^{C_{i}^{2}} - (i-1)!*\sum\limits_{j=1}^{i}\frac{f_{j}}{(j-1)!} * \frac{g_{i-j}}{(i-j)!} \)
所以把 \( g[i] \) 的定义改成 \( g[i] = \frac{2^{C_{i}^{2}}}{i!} \)
于是 \( f[i] = 2^{C_{i}^{2}} - (i-1)!*\sum\limits_{j=1}^{i}\frac{f_{j}}{(j-1)!} * g_{i-j} \)
然后就可以分治FFT了;
很容易写错的地方是那个 \( 2^{C_{i}^{2}} \),因为是指数,所以应该对 \( mod-1 \) 取模,而不是 \( mod \) !!
而且除以 \( 2 \) 也不能预处理 \( inv2 \) 了,因为 \( mod-1 \) 不是质数!!
代码如下:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
int const xn=(<<),mod=;
int n,f[xn],g[xn],jc[xn],jcn[xn],a[xn],b[xn],rev[xn],inv2,in[xn];
int rd()
{
int ret=,f=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=; ch=getchar();}
while(ch>=''&&ch<='')ret=ret*+ch-'',ch=getchar();
return f?ret:-ret;
}
ll pw(ll a,int b,int p=mod)
{
ll ret=;
for(;b;b>>=,a=(a*a)%p)if(b&)ret=(ret*a)%p;
return ret;
}
int upt(int x){while(x>=mod)x-=mod; while(x<)x+=mod; return x;}
int C(int x){return ((ll)x*(x-)/)%(mod-);}//mod-1!!! //not inv2 -- !pri
void init()
{
jc[]=;
for(int i=;i<=n;i++)jc[i]=(ll)jc[i-]*i%mod;
jcn[n]=pw(jc[n],mod-);
for(int i=n-;i>=;i--)jcn[i]=(ll)jcn[i+]*(i+)%mod;
g[]=;
for(int i=;i<=n;i++)g[i]=(ll)pw(,C(i))*jcn[i]%mod;
}
void ntt(int *a,int tp,int lim)
{
for(int i=;i<lim;i++)
if(i<rev[i])swap(a[i],a[rev[i]]);
for(int mid=;mid<lim;mid<<=)
{
int len=(mid<<),wn=pw(,tp==?(mod-)/len:(mod-)-(mod-)/len);
for(int j=;j<lim;j+=len)
for(int k=,w=;k<mid;k++,w=(ll)w*wn%mod)
{
int x=a[j+k],y=(ll)w*a[j+mid+k]%mod;
a[j+k]=upt(x+y); a[j+mid+k]=upt(x-y);
}
}
if(tp==)return; int inv=pw(lim,mod-);
for(int i=;i<lim;i++)a[i]=(ll)a[i]*inv%mod;
}
void work(int l,int r)
{
if(l==r){f[l]=upt((ll)pw(,C(l))-(ll)jc[l-]*f[l]%mod); return;}
int len=r-l+,mid=((l+r)>>);
work(l,mid);
int lim=,L=;
while(lim<=len)lim<<=,L++;//
for(int i=;i<lim;i++)rev[i]=((rev[i>>]>>)|((i&)<<(L-)));
for(int i=;i<lim;i++)a[i]=b[i]=;
for(int i=l;i<=mid;i++)a[i-l]=(ll)f[i]*jcn[i-]%mod;//jcn!!
for(int i=;i<=r-l;i++)b[i]=g[i];//
ntt(a,,lim); ntt(b,,lim);
for(int i=;i<lim;i++)a[i]=(ll)a[i]*b[i]%mod;
ntt(a,-,lim);
for(int i=mid+;i<=r;i++)f[i]=upt(f[i]+a[i-l]);
work(mid+,r);
}
int main()
{
n=rd(); init();
work(,n);
printf("%d\n",f[n]);
return ;
}
分治FFT
也可以用多项式求逆做:
设 \( g[i] = 2^{C_{i}^{2}} \),这里 \( g[0] = 1 \),\( f[i] \) 定义同上;
得到 \( g[n] = \sum\limits_{i=1}^{n} C_{n-1}^{i-1} * f[i] * g[n-i] \)
拆 \( C_{n-1}^{i-1} \),得到
\( \frac{g[n]}{(n-1)!} = \sum\limits_{i=1}^{n} \frac{f[i]}{(i-1)!} * \frac{g[n-i]}{(n-i)!} \)
设 \( A[i] = \frac{g[i]}{(i-1)!} \) , \( B[i] = \frac{f[i]}{(i-1)!} \) , \( C[i] = \frac{g[i]}{i!} \)
注意这里 \( A[0] = 0 \) , \( B[0] = 0 \) , \( C[0] = 1 \)
所以 \( A(x) = B(x) * C(x) \)
即 \( B(x) = A(x) * C^{-1}(x) ( mod x^{n+1} ) \)
多项式求逆即可;
别忘了外层的乘法还要处理一下 \( rev \) 数组。
代码如下:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
int const xn=(<<),mod=;
int n,a[xn],b[xn],c[xn],t[xn],rev[xn],jc[xn],jcn[xn];
int rd()
{
int ret=,f=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=; ch=getchar();}
while(ch>=''&&ch<='')ret=ret*+ch-'',ch=getchar();
return f?ret:-ret;
}
ll pw(ll a,int b)
{
ll ret=;
for(;b;b>>=,a=(a*a)%mod)if(b&)ret=(ret*a)%mod;
return ret;
}
int upt(int x){while(x>=mod)x-=mod; while(x<)x+=mod; return x;}
int C(int x){return ((ll)x*(x-)/)%(mod-);}
void init()
{
jc[]=;
for(int i=;i<=n;i++)jc[i]=(ll)jc[i-]*i%mod;
jcn[n]=pw(jc[n],mod-);
for(int i=n-;i>=;i--)jcn[i]=(ll)jcn[i+]*(i+)%mod;
}
void ntt(int *a,int tp,int lim)
{
for(int i=;i<lim;i++)
if(i<rev[i])swap(a[i],a[rev[i]]);
for(int mid=;mid<lim;mid<<=)
{
int len=(mid<<),wn=pw(,tp==?(mod-)/len:(mod-)-(mod-)/len);
for(int j=;j<lim;j+=len)
for(int k=,w=;k<mid;k++,w=(ll)w*wn%mod)
{
int x=a[j+k],y=(ll)w*a[j+mid+k]%mod;
a[j+k]=upt(x+y); a[j+mid+k]=upt(x-y);
}
}
if(tp==)return; int inv=pw(lim,mod-);
for(int i=;i<lim;i++)a[i]=(ll)a[i]*inv%mod;
}
void inv(int *a,int *b,int n)
{
if(n==){b[]=pw(a[],mod-); return;}
inv(a,b,(n+)>>);
int lim=,l=;
while(lim<n+n)lim<<=,l++;
for(int i=;i<lim;i++)rev[i]=((rev[i>>]>>)|((i&)<<(l-)));
for(int i=;i<n;i++)t[i]=a[i];
for(int i=n;i<lim;i++)t[i]=;
ntt(t,,lim); ntt(b,,lim);
for(int i=;i<lim;i++)b[i]=upt((-(ll)t[i]*b[i])%mod*b[i]%mod);
ntt(b,-,lim);
for(int i=n;i<lim;i++)b[i]=;
}
int main()
{
n=rd(); init();
int lim=,l=;
while(lim<=n)lim<<=,l++;
for(int i=;i<lim;i++)a[i]=(ll)pw(,C(i))*jcn[i-]%mod;
for(int i=;i<lim;i++)c[i]=(ll)pw(,C(i))*jcn[i]%mod;
c[]=;
inv(c,b,n+);
for(int i=;i<lim;i++)rev[i]=((rev[i>>]>>)|((i&)<<(l-)));//!!!
ntt(a,,lim); ntt(b,,lim);
for(int i=;i<lim;i++)b[i]=(ll)a[i]*b[i]%mod;
ntt(b,-,lim);
printf("%lld\n",(ll)b[n]*jc[n-]%mod);
return ;
}
多项式求逆
也可以用指数型生成函数,具体做法可以看这篇博客:https://blog.csdn.net/wzq_qwq/article/details/48435621
关于为什么 \( G(x) = e^{F(x)} \)
因为从组合意义上来看,任意无向图实际上可以分为:由0个连通图组成(没有点),由1个连通图组成,由2个连通图组成(这2个之间不连通),由3个连通图组成...
所以 \( G(x) = 1 + \frac{F(x)}{1!} + \frac{F(x)^{2}}{2!} + \frac{F(x)^{3}}{3!} + ... \)
即 \( G(x) = e^{F(x)} \)
而 \( G(x) \) 很好构造,求 \( F(x) = lnG(x) \)
求 \( ln \) 的方法可以看这两篇博客:
https://blog.csdn.net/litble/article/details/81749788
https://blog.csdn.net/ezoixx118/article/details/81235586
直接用上公式...这题原来是多项式求 \( ln \) 的裸题;
别忘了最后乘上 \( n! \)
代码如下:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
int const xn=(<<),mod=;
int n,g[xn],dg[xn],ig[xn],t[xn],rev[xn],jc[xn],jcn[xn];
int rd()
{
int ret=,f=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=; ch=getchar();}
while(ch>=''&&ch<='')ret=ret*+ch-'',ch=getchar();
return f?ret:-ret;
}
ll pw(ll a,ll b)
{
ll ret=; b=b%(mod-);
for(;b;b>>=,a=(a*a)%mod)if(b&)ret=(ret*a)%mod;
return ret;
}
int upt(int x){while(x>=mod)x-=mod; while(x<)x+=mod; return x;}
void init()
{
jc[]=;
for(int i=;i<=n;i++)jc[i]=(ll)jc[i-]*i%mod;
jcn[n]=pw(jc[n],mod-);
for(int i=n-;i>=;i--)jcn[i]=(ll)jcn[i+]*(i+)%mod;
}
void ntt(int *a,int tp,int lim)
{
for(int i=;i<lim;i++)
if(i<rev[i])swap(a[i],a[rev[i]]);
for(int mid=;mid<lim;mid<<=)
{
int len=(mid<<),wn=pw(,tp==?(mod-)/len:(mod-)-(mod-)/len);
for(int j=;j<lim;j+=len)
for(int k=,w=;k<mid;k++,w=(ll)w*wn%mod)
{
int x=a[j+k],y=(ll)w*a[j+mid+k]%mod;
a[j+k]=upt(x+y); a[j+mid+k]=upt(x-y);
}
}
if(tp==)return; int inv=pw(lim,mod-);
for(int i=;i<lim;i++)a[i]=(ll)a[i]*inv%mod;
}
void inv(int *a,int *b,int n)
{
if(n==){b[]=pw(a[],mod-); return;}
inv(a,b,(n+)>>);
int lim=,l=;
while(lim<n+n)lim<<=,l++;
for(int i=;i<lim;i++)rev[i]=((rev[i>>]>>)|((i&)<<(l-)));
for(int i=;i<n;i++)t[i]=a[i];
for(int i=n;i<lim;i++)t[i]=;
ntt(t,,lim); ntt(b,,lim);
for(int i=;i<lim;i++)b[i]=upt((-(ll)t[i]*b[i])%mod*b[i]%mod);
ntt(b,-,lim);
for(int i=n;i<lim;i++)b[i]=;
}
int main()
{
n=rd(); init();
for(int i=;i<=n;i++)
{
if(i<)g[i]=;
g[i]=(ll)pw(,(ll)i*(i-)/)*jcn[i]%mod;
}
for(int i=;i<=n;i++)dg[i-]=(ll)i*g[i]%mod;
dg[n]=;//
inv(g,ig,n+);
int lim=,l=;
while(lim<n+n)lim<<=,l++;
for(int i=;i<lim;i++)rev[i]=((rev[i>>]>>)|((i&)<<(l-)));
ntt(ig,,lim); ntt(dg,,lim);
for(int i=;i<lim;i++)dg[i]=(ll)ig[i]*dg[i]%mod;
ntt(dg,-,lim);
printf("%lld\n",(ll)dg[n-]*pw(n,mod-)%mod*jc[n]%mod);
return ;
}
多项式求ln
bzoj 3456 城市规划 —— 分治FFT / 多项式求逆 / 指数型生成函数(多项式求ln)的更多相关文章
- bzoj 3456 城市规划——分治FFT / 多项式求逆 / 多项式求ln
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3456 分治FFT: 设 dp[ i ] 表示 i 个点时连通的方案数. 考虑算补集:连通的方 ...
- [BZOJ 3456]城市规划(cdq分治+FFT)
[BZOJ 3456]城市规划(cdq分治+FFT) 题面 求有标号n个点无向连通图数目. 分析 设\(f(i)\)表示\(i\)个点组成的无向连通图数量,\(g(i)\)表示\(i\)个点的图的数量 ...
- BZOJ 3456: 城市规划 与 多项式求逆算法介绍(多项式求逆, dp)
题面 求有 \(n\) 个点的无向有标号连通图个数 . \((1 \le n \le 1.3 * 10^5)\) 题解 首先考虑 dp ... 直接算可行的方案数 , 容易算重复 . 我们用总方案数减 ...
- bzoj 3456 城市规划 无向简单连通图个数 多项式求逆
题目大意 求n个点的无向简单连通图个数 做法1 \(f[i]\)表示i个点的无向简单连通图个数 \(g[i]=2^{\frac {i*(i-1)}{2}}\)表示i个点的无向简单图个数(不要求连通) ...
- BZOJ 3456 城市规划 ( NTT + 多项式求逆 )
题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=3456 题意: 求出\(n\)个点的简单(无重边无自环)无向连通图的个数.(\(n< ...
- BZOJ 3456: 城市规划 多项式求逆
Description 刚刚解决完电力网络的问题, 阿狸又被领导的任务给难住了. 刚才说过, 阿狸的国家有n个城市, 现在国家需要在某些城市对之间建立一些贸易路线, 使得整个国家的任意两个城市都直接 ...
- BZOJ 3456: 城市规划(dp+多项式求逆)
传送门 解题思路 这道题就是求带标号的无向连通图个数,首先考虑\(O(n^2)\)的做法,设\(f_i\)表示有\(i\)个节点的无向连通图个数,那么考虑容斥,先把所有的无向图求出,即为\(2^{C( ...
- BZOJ 3456: 城市规划 [多项式求逆元 组合数学 | 生成函数 多项式求ln]
3456: 城市规划 题意:n个点组成的无向连通图个数 以前做过,今天复习一下 令\(f[n]\)为n个点的无向连通图个数 n个点的完全图个数为\(2^{\binom{n}{2}}\) 和Bell数的 ...
- BZOJ 3456 城市规划 (组合计数、DP、FFT)
题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=3456 著名的多项式练习题,做法也很多,终于切掉了纪念 首先求一波递推式: 令\(F(n ...
随机推荐
- Xenomai for Debian Jessie
安装内核源码包 apt install linux-source-3.16 安装其他编译需要的工具: apt install build-essential libc-dev libc6-dev pk ...
- 让uboot的tftp支持上传功能
转载:http://blog.chinaunix.net/uid-20737871-id-2124122.html uboot下的tftp下载功能是非常重要和常见的功能.但是偶尔有些特殊需求的人需要使 ...
- Nginx 经验小结
chmod 777 永远不要 使用 777,有时候可以懒惰的解决权限问题, 但是它同样也表示你没有线索去解决权限问题,你只是在碰运气. 你应该检查整个路径的权限,并思考发生了什么事情. 把 root ...
- 3_Jsp标签_简单标签_防盗链和转义标签的实现
一概念 1防盗链 在HTTP协议中,有一个表头字段叫referer,采用URL的格式来表示从哪儿链接到当前的网页或文件,通过referer,网站可以检测目标网页访问的来源网页.有了referer跟踪来 ...
- 【BZOJ4966】总统选举 线段树+随机化
[BZOJ4966]总统选举 Description 黑恶势力的反攻计划被小C成功摧毁,黑恶势力只好投降.秋之国的人民解放了,举国欢庆.此时,原秋之国总统因没能守护好国土,申请辞职,并请秋之国人民的大 ...
- 【剑指Offer学习】【面试题33:把数组排成最小的数】
题目:输入一个正整数数组,把数组里全部数字拼接起来排成一个数.打印能拼接出的全部数字中最小的一个. 样例说明: 比如输入数组{3. 32, 321},则扫描输出这3 个数字能排成的最小数字321323 ...
- MySQL修改配置 区分大小写
在使用mysql的时候,数据库名,表名,字段名等有大小写的区分,这个可以通过配置文件设置.如果设置了严格区分大小写,在访问表的时候没有注意到表名的大小写,将会报出表不存在的错误.下面是修改配置文件: ...
- view上下抖动特效
shake.xml <?xml version="1.0" encoding="utf-8"?> <translate xmlns:andro ...
- node版本管理工具 -- nvm安装与使用
新老项目维护时node环境切换麻烦怎么办? 不用担心,有了nvm ,一个命令就能切换node版本. 首先需要安装nvm工具,进入下载地址. 下载之后安装nvm. nvm安装之后还需要配置两个环境变量( ...
- 使用vscode写typescript(node.js环境)起手式
动机 一直想把typescript在服务端开发中用起来,主要原因有: javascript很灵活,但记忆力不好的话,的确会让你头疼,看着一月前自己写的代码,一脸茫然. 类型检查有利有敝,但在团队开发中 ...