FFT与NTT的模板
网上相关博客不少,这里给自己留个带点注释的模板,以后要是忘了作提醒用。
以洛谷3803多项式乘法裸题为例。
FFT:
#include <cstdio>
#include <cmath>
#include <cctype>
#include <algorithm>
#define ri readint()
#define gc getchar() int readint() {
int x = , s = , c = gc;
while (c <= ) c = gc;
if (c == '-') s = -, c = gc;
for (; isdigit(c); c = gc) x = x * + c - ;
return x * s;
} const int maxn = * 1e6 + ;
const double PI = acos(-1.0); struct Complex {
double x, y;
Complex(double a = , double b = ):x(a), y(b){}
};
Complex operator + (Complex A, Complex B) { return Complex(A.x + B.x, A.y + B.y); }
Complex operator - (Complex A, Complex B) { return Complex(A.x - B.x, A.y - B.y); }
Complex operator * (Complex A, Complex B) { return Complex(A.x * B.x - A.y * B.y, A.x * B.y + A.y * B.x); } Complex a[maxn], b[maxn];
int n, m;
int r[maxn], l, limit = ; void fft(Complex *A, int type) {
for (int i = ; i < limit; i++)
if (i < r[i])
std::swap(A[i], A[r[i]]);
//迭代方式模拟递归写法,需要理解递归是怎么做的才能看懂这个
for (int mid = ; mid < limit; mid <<= ) {
//本来单位根是2*PI/len,这里len替换成2*mid,2就约掉了
Complex Wn(cos(PI / mid), type * sin(PI / mid));
for (int R = mid << , j = ; j < limit; j += R) {
Complex w(, );//单位根的k次幂
for (int k = ; k < mid; k++, w = w * Wn) {
//蝴蝶变换
Complex x = A[j+k], y = w * A[j+k+mid];
A[j+k] = x + y;
A[j+k+mid] = x - y;
}
}
}
} int main() {
n = ri, m = ri;
for (int i = ; i <= n; i++)
a[i].x = ri;
for (int i = ; i <= m; i++)
b[i].x = ri; while (limit <= n + m) {//长度变为2^l
limit <<= ;
l++;
}
for (int i = ; i < limit; i++)//二进制镜像
r[i] = (r[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));
return ;
}
NTT是用模域取代了复数域,性质相同只是换了单位根,所以板子基本相同。我这两个相比NTT确实比FFT快一点的:
#include <bits/stdc++.h>
#define ll long long
#define ri readll()
#define gc getchar()
#define rep(i, a, b) for (int i = a; i <= b; i++)
using namespace std; const int P = , G = , Gi = , maxn = * 1e6 + ;
//P的原根为3,3%P的逆元为332748118
//原根意味着:3^(P-1) % P = 1,其中P-1是3%P的阶,本应是φ(P),这里恰好为大素数
ll n, m;
ll a[maxn], b[maxn];
int limit = , l, r[maxn]; ll readll() {
ll x = 0ll, s = 1ll;
char c = gc;
while (c <= ) c = gc;
if (c == '-') s = -1ll, c = gc;
for (; isdigit(c); c = gc) x = x * + c - ;
return x * s;
} ll ksm(ll a, ll b, int mod) {
ll res = 1ll;
for (; b; b >>= ) {
if (b & ) res = res * a % mod;
a = a * a % mod;
}
return res;
} void NTT(ll *A, int flag) {
rep(i, , limit)
if (i < r[i])
swap(A[i], A[r[i]]); for (int mid = ; mid < limit; mid <<= ) {
//如果是变换则单位根为3^[(P-1)/(len)] % P,逆变换则用逆元
ll Wn = ksm(flag ? G : Gi, (P-) / (mid*), P);
for (int R = mid << , j = ; j < limit; j += R) {
ll w = 1ll;
for (int k = ; k < mid; k++, w = w * Wn % P) {
ll x = A[j+k], y = A[j+k+mid] * w % P;
A[j+k] = (x + y) % P;
A[j+k+mid] = (x - y + P) % P;
}
}
}
} int main() {
n = ri, m = ri;
rep(i, , n) a[i] = (ri + P) % P;
rep(i, , m) b[i] = (ri + P) % P; while (limit < n + m + ) {
limit <<= ;
l++;
}
rep(i, , limit) r[i] = (r[i>>] >> ) | ((i & ) << (l - ));
NTT(a, ); NTT(b, );
rep(i, , limit) a[i] = a[i] * b[i] % P;
NTT(a, ); ll inv = ksm(limit, P - , P);//最后变换回来要乘长度的逆元
rep(i, , n + m) printf("%lld ", a[i] * inv % P); return ;
}
FFT与NTT的模板的更多相关文章
- 多项式乘法,FFT与NTT
多项式: 多项式?不会 多项式加法: 同类项系数相加: 多项式乘法: A*B=C $A=a_0x^0+a_1x^1+a_2x^2+...+a_ix^i+...+a_{n-1}x^{n-1}$ $B=b ...
- FFT,NTT 专题
学习傅里叶的基本性质及其代码,可以参考大神理解 还有 ACdream 的博客 贴一下NTT的模板: using namespace std; typedef long long ll; int n; ...
- FFT和NTT学习笔记_基础
FFT和NTT学习笔记 算法导论 参考(贺) http://picks.logdown.com/posts/177631-fast-fourier-transform https://blog.csd ...
- fft,ntt总结
一个套路:把式子推成卷积形式,然后用fft或ntt优化求解过程. fft的扩展性不强,不可以在fft函数里多加骚操作--DeepinC T1:多项式乘法 板子题 T2:快速傅立叶之二 另一个板子,小技 ...
- 多项式fft、ntt、fwt 总结
做了四五天的专题,但是并没有刷下多少题.可能一开始就对多项式这块十分困扰,很多细节理解不深. 最简单的形式就是直接两个多项式相乘,也就是多项式卷积,式子是$N^2$的.多项式算法的过程就是把卷积做一种 ...
- FFT与NTT专题
先不管旋转操作,考虑化简这个差异值 $$begin{aligned}sum_{i=1}^n(x_i-y_i-c)^2&=sum_{i=1}^n(x_i-y_i)^2+nc^2-2csum_{i ...
- 洛谷 - P3803 -【模板】多项式乘法(FFT) - NTT
https://www.luogu.org/problemnew/show/P3803 看别人偏偏就是要用NTT去过.实验证明大概是这样用.求0~n的多项式和0~m的多项式的乘积.注意MAXN取值.A ...
- luoguP4721 【模板】分治 FFT (分治NTT)
给定 $g[1....n-1]$,求 $f[0],f[1],...,f[n-1]$,其中 $f[i]=\sum_{j=1}^{i}f[i-j]g[j]$ 变界为 $f[0]=1$ 答案模 9 ...
- 卷积FFT、NTT、FWT
先简短几句话说说FFT.... 多项式可用系数和点值表示,n个点可确定一个次数小于n的多项式. 多项式乘积为 f(x)*g(x),显然若已知f(x), g(x)的点值,O(n)可求得多项式乘积的点值. ...
随机推荐
- Vue实例和方法
github地址:https://github.com/manlili/vue_learn里面的lesson03 一 实例 每个 Vue 实例都会代理其 data 对象里所有的属性,改变data,vu ...
- UVA11149 Power of Matrix —— 矩阵倍增、矩阵快速幂
题目链接:https://vjudge.net/problem/UVA-11149 题意: 给出矩阵A,求出A^1 + A^2 …… + A^k . 题解: 1.可知:A^1 + A^2 …… + A ...
- Nginx中的惊群现象解决方法
*什么是惊群现象?Nginx中用了什么方法来避免这种问题的发生?本篇就解决这两个问题...→_→* 惊群现象的定义与危害 在Nginx中,每一个worker进程都是由master进程fork出来的.m ...
- 关于lock锁
在 jdk1.5 之后,并发包中新增了 Lock 接口(以及相关实现类)用来实现锁功能,Lock 接口提供了与 synchronized 关键字类似的同步功能,但需要在使用时手动获取锁和释放锁. lo ...
- 小程序observer函数的应用
需求是这样的 就是构建月份的组件中,月份小于10月的时候 显示的数字都是一个位数,需要转换成两位数, 比如8月份是8 ,那就要转换为08 ,同理可得 其他低于十月份的月份也是要这样做: 打开组件的js ...
- ICE协议下NAT穿越的实现(STUN&TURN)
正文: 一. 首先来简单讲讲什么是NAT? 原来这是因为IPV4引起的,我们上网很可能会处在一个NAT设备(无线路由器之类)之后.NAT设备会在IP封包通过设备时修改源/目的IP地址. 对于家用路由器 ...
- gsoap开发webservice
gSOAP编译工具提供了一个SOAP/XML 关于C/C++ 语言的实现,从而让C/C++语言开发web服务或客户端程序的工作变得轻松了很多.绝大多数的C++web服务工具包提供一组API函数类库来处 ...
- Dynamic Gcd
树链剖分+差分 直接区间加显然是不行的,由于gcd(a,b,c)=gcd(a,a-b,b-c),那么我们对这些数差分,然后就变成单点修改.原本以为这道题很简单,没想到这么麻烦,就膜了发代码. 首先我们 ...
- 堆栈(栈stack)的实现和基本用法(二)
个人网站http://www.ravedonut.com/ 栈的应用: #include <iostream> #include <stack> using namespace ...
- 1.6-1.10 使用Sqoop导入数据到HDFS及一些设置
一.导数据 1.import和export Sqoop可以在HDFS/Hive和关系型数据库之间进行数据的导入导出,其中主要使用了import和export这两个工具.这两个工具非常强大, 提供了很多 ...