FFT
 #include<iostream>
#include<cstring>
#include<cstdlib>
#include<cstdio>
#include<cmath>
#include<algorithm>
#define maxn 1000005
using namespace std;
inline int read() {
int x=,f=;char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-;
for(;isdigit(ch);ch=getchar()) x=x*+ch-'';
return x*f;
}
double pi=acos(-1.0);
struct complex {
double x,y;
complex (double xx=,double yy=) {x=xx;y=yy;}
complex operator +(const complex b) const {return complex(x+b.x,y+b.y);}
complex operator -(const complex b) const {return complex(x-b.x,y-b.y);}
complex operator *(const complex b) const {return complex(x*b.x-y*b.y,x*b.y+y*b.x);}
}a[maxn],b[maxn];
int n,m;
int limit=,l,pos[maxn];
void FFT(complex *A,int tp) {
for(int i=;i<limit;i++) if(i<pos[i]) swap(A[i],A[pos[i]]);
for(int mid=;mid<limit;mid<<=) {
complex wn(cos(pi/mid),tp*sin(pi/mid));
for(int R=mid<<,j=;j<limit;j+=R) {
complex w(,);
for(int k=;k<mid;k++,w=w*wn) {
complex x=A[j+k],y=w*A[j+mid+k];
A[j+k]=x+y;
A[j+mid+k]=x-y;
}
}
}
return ;
}
int main() {
n=read(),m=read();
for(int i=;i<=n;i++) a[i].x=read();
for(int i=;i<=m;i++) b[i].x=read();
while(limit<=n+m) limit<<=,l++;
for(int i=;i<limit;i++) pos[i]=(pos[i>>]>>)|((i&)<<(l-));
FFT(a,);
FFT(b,);
for(int i=;i<limit;i++) a[i]=a[i]*b[i];
FFT(a,-);
for(int i=;i<=n+m;i++) printf("%d ",(int)(a[i].x/limit+0.5));
}

FFT

 生成函数

小A有ai个价值为Ai的物品,小B有bi个价值为Ai的物品,求用两个组成价值为ci的方案数

生成函数可以解决上面的这个问题,构造两个多项式,第X的Ai次方项的系数表示价值为i的物品有多少个,对两个人分别构造,乘在一起的多项式就代表所有的方案数。

原根

定义P的原根为满足的整数g。

NTT

 #include<iostream>
#include<cstring>
#include<cstdlib>
#include<cstdio>
#include<cmath>
#include<algorithm>
#define maxn 4000001
#define mod 998244353
#define ll long long
using namespace std;
inline int read() {
int x=,f=;char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-;
for(;isdigit(ch);ch=getchar()) x=x*+ch-'';
return x*f;
}
ll a[maxn],b[maxn],pos[maxn];
ll n,m,limit=,l,g=;
ll power(ll x,ll y) {
ll ans=;
while(y) {
if(y&) ans*=x,ans%=mod;
x*=x,x%=mod;y>>=;
}
return ans;
}
void NTT(ll *A,int tp) {
for(int i=;i<limit;i++) if(i<pos[i]) swap(A[i],A[pos[i]]);
for(int mid=;mid<limit;mid<<=) {
ll wn=power(g,(mod-)/(mid<<));
if(tp==-) wn=power(wn,mod-);
for(int j=;j<limit;j+=(mid<<)) {
ll w=;
for(int k=;k<mid;k++,w*=wn,w%=mod) {
ll x=A[j+k],y=w*A[j+mid+k];
A[j+k]=x+y;A[j+k]=(A[j+k]%mod+mod)%mod;
A[j+mid+k]=x-y;A[j+mid+k]=(A[j+mid+k]%mod+mod)%mod;
}
}
}
if(tp==-) {
ll ny=power(limit,mod-);
for(int i=;i<limit;i++) A[i]*=ny,A[i]=(A[i]%mod+mod)%mod;
}
}
int main() {
n=read(),m=read();
for(int i=;i<=n;i++) a[i]=read();
for(int i=;i<=m;i++) b[i]=read();
while(limit<=n+m) limit<<=,l++;
for(int i=;i<limit;i++) pos[i]=(pos[i>>]>>)|((i&)<<(l-));
NTT(a,);
NTT(b,);
for(int i=;i<limit;i++) a[i]=a[i]*b[i],a[i]=(a[i]%mod+mod)%mod;
NTT(a,-);
for(int i=;i<=n+m;i++) printf("%lld ",a[i]);
}

NTT

多项式求逆

 #include<iostream>
#include<cstring>
#include<cstdlib>
#include<cstdio>
#include<cmath>
#include<algorithm>
#define ll long long
#define mod 998244353
#define maxn 1000000
using namespace std;
inline int read() {
int x=,f=;char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-;
for(;isdigit(ch);ch=getchar()) x=x*+ch-'';
return x*f;
}
ll g=,limit=,l,n;
ll a[maxn],b[maxn],pos[maxn],c[maxn];
ll power(ll x,ll y) {
ll ans=;
while(y) {
if(y&) ans*=x,ans%=mod;
x*=x,x%=mod;y>>=;
}
return ans;
}
void NTT(ll *A,int tp) {
for(int i=;i<limit;i++) if(i<pos[i]) swap(A[i],A[pos[i]]);
for(int mid=;mid<limit;mid<<=) {
ll wn=power(g,(mod-)/(mid<<));
if(tp==-) wn=power(wn,mod-);
for(int j=;j<limit;j+=(mid<<)) {
ll w=;
for(int k=;k<mid;k++,w*=wn,w%=mod) {
ll x=A[j+k],y=w*A[j+mid+k]%mod;
A[j+k]=x+y;A[j+k]=(A[j+k]%mod+mod)%mod;
A[j+mid+k]=x-y;A[j+mid+k]=(A[j+mid+k]%mod+mod)%mod;
}
}
}
if(tp==-) {
ll ny=power(limit,mod-);
for(int i=;i<limit;i++) A[i]*=ny,A[i]=(A[i]%mod+mod)%mod;
}
}
int d[maxn];
void inv(int step,ll *A,ll *B) {
if(step==) {B[]=power(A[],mod-);return;}
inv((step+)>>,A,B);
l=,limit=;
while(limit<=(step<<)) limit<<=,l++;
for(int i=;i<limit;i++) pos[i]=(pos[i>>]>>)|((i&)<<(l-));
for(int i=;i<step;i++) c[i]=A[i];
for(int i=step;i<limit;i++) c[i]=;
NTT(c,);NTT(B,);
for(int i=;i<limit;i++) B[i]=((2ll-c[i]*B[i]%mod)+mod)%mod*B[i]%mod;
NTT(B,-);
for(int i=step;i<limit;i++) B[i]=;
}
int main() {
n=read();
for(int i=;i<n;i++) a[i]=read();
inv(n,a,b);
for(int i=;i<n;i++) printf("%lld ",b[i]);
}

多项式求逆

多项式开根

 void getsqr(ll *A,ll *B,ll len) {
if(len==) {B[]=;return;}
getsqr(A,B,(len+)>>);
memset(invb,,sizeof(invb));
getinv(B,invb,len);
l=,limit=;
while(limit<=(len<<)) limit<<=,l++;
for(int i=;i<limit;i++) pos[i]=(pos[i>>]>>)|((i&)<<(l-));
for(int i=;i<len;i++) c[i]=A[i];
for(int i=len;i<limit;i++) c[i]=;
NTT(invb,);NTT(c,);NTT(B,);
for(int i=;i<limit;i++) {B[i]=((c[i]*invb[i]%mod)*inv2%mod+B[i]*inv2)%mod;}
NTT(B,-);
for(int i=len;i<limit;i++) B[i]=;
}

FFT模板 生成函数 原根 多项式求逆 多项式开根的更多相关文章

  1. 【BZOJ3456】轩辕朗的城市规划 无向连通图计数 CDQ分治 FFT 多项式求逆 多项式ln

    题解 分治FFT 设\(f_i\)为\(i\)个点组成的无向图个数,\(g_i\)为\(i\)个点组成的无向连通图个数 经过简单的推导(枚举\(1\)所在的连通块大小),有: \[ f_i=2^{\f ...

  2. bzoj 3456 城市规划——分治FFT / 多项式求逆 / 多项式求ln

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3456 分治FFT: 设 dp[ i ] 表示 i 个点时连通的方案数. 考虑算补集:连通的方 ...

  3. 2019.01.01 bzoj3625:小朋友和二叉树(生成函数+多项式求逆+多项式开方)

    传送门 codeforces传送门codeforces传送门codeforces传送门 生成函数好题. 卡场差评至今未过 题意简述:nnn个点的二叉树,每个点的权值KaTeX parse error: ...

  4. 【BZOJ3625】【codeforces438E】小朋友和二叉树 生成函数+多项式求逆+多项式开根

    首先,我们构造一个函数$G(x)$,若存在$k∈C$,则$[x^k]G(x)=1$. 不妨设$F(x)$为最终答案的生成函数,则$[x^n]F(x)$即为权值为$n$的神犇二叉树个数. 不难推导出,$ ...

  5. NTT+多项式求逆+多项式开方(BZOJ3625)

    定义多项式$h(x)$的每一项系数$h_i$,为i在c[1]~c[n]中的出现次数. 定义多项式$f(x)$的每一项系数$f_i$,为权值为i的方案数. 通过简单的分析我们可以发现:$f(x)=\fr ...

  6. 【learning】多项式相关(求逆、开根、除法、取模)

    (首先要%miskcoo,这位dalao写的博客(这里)实在是太强啦qwq大部分多项式相关的知识都是从这位dalao博客里面学的,下面这篇东西是自己对其博客学习后的一些总结和想法,大部分是按照其博客里 ...

  7. [Codeforces438E][bzoj3625] 小朋友和二叉树 [多项式求逆+多项式开根]

    题面 传送门 思路 首先,我们把这个输入的点的生成函数搞出来: $C=\sum_{i=0}^{lim}s_ix^i$ 其中$lim$为集合里面出现过的最大的数,$s_i$表示大小为$i$的数是否出现过 ...

  8. P6295-有标号 DAG 计数【多项式求逆,多项式ln】

    正题 题目链接:https://www.luogu.com.cn/problem/P6295 题目大意 求所有\(n\)个点的弱联通\(DAG\)数量. \(1\leq n\leq 10^5\) 解题 ...

  9. bzoj 3625小朋友和二叉树 多项式求逆+多项式开根 好题

    题目大意 给定n种权值 给定m \(F_i表示权值和为i的二叉树个数\) 求\(F_1,F_2...F_m\) 分析 安利博客 \(F_d=F_L*F_R*C_{mid},L+mid+R=d\) \( ...

随机推荐

  1. STL使用总结

    转载于http://blog.csdn.net/daisy_chenting/article/details/6898184 1.    概述 泛型编程思想最早缘于A.Stepanov提出的部分算法可 ...

  2. Balanced Sequence(毒瘤啊)排序贪心 HDU多校

    Problem Description Chiaki has n strings s1,s2,…,sn consisting of '(' and ')'. A string of this type ...

  3. stout代码分析之五:UUID类

    UUID全称通用唯一识别码,被广泛应用于分布式系统中,让所有的元素具有唯一的标识. stout中UUID类继承自boost::uuids::uuid.api如下: random, 产生一个UUID对象 ...

  4. 在Linux防火墙上过滤外来的ICMP timestamp

    ICMP timestamp请求响应漏洞 解决方案:  * 在您的防火墙上过滤外来的ICMP timestamp(类型13)报文以及外出的ICMP timestamp回复报文.     具体解决方式就 ...

  5. 用好printf和scanf

    转载自:http://hi.baidu.com/wuxicn/item/f648fe1970f86917e3f98682 在C中,printf系列函数(fprintf, sprintf...)和sca ...

  6. ES6中字符串的扩展

    一.查找字符串 在ES5中,可以使用 indexOf 方法和 lastIndexOf 方法查找字符串: let str = 'hello world'; alert(str.indexOf('o')) ...

  7. 本地更新代码同步至github仓库

    昨晚在家里写了一个demo放到github上,然后今天晚上来公司准备搞一下,但是git pull下来在本地修改之后push不到github上,然后发现公司电脑上并没有access权限,然后想起来还没配 ...

  8. C# 后台获取请求来源、文件下载

    文件流下载文件 void BigFileDownload() { try { string FileName = "测试.docx"; string filePath = Page ...

  9. spring和Quartz的集群(二)

    一:前沿 写完了这两篇才突然想起来,忘记了最关键的东西,那就是在配置文件这里的配置,还有数据库的配置.这是郁闷啊!继续吧! 二:内容配置 我们在集成的时候需要自己配置一个quartz.properti ...

  10. Vue.js -- 过滤器

    VueJs中的过滤器基础 过滤器是一个通过输入数据,能够及时对数据进行处理并返回一个数据结果的简单函数.Vue有很多很便利的过滤器,可以参考官方文档, http://cn.vuejs.org/api/ ...