FFT是什么?

FFT是快速傅里叶变换(fast Fourier transform)的简称。在ACM领域主要是用来快速求解多项式乘法的算法,

在信号领域也有很大用途


基础知识

  1. 卷积

    举个例子,给你两个向量 \(a (a_0, a_1, a_2), b(b_0, b_1, b_2)\)

    a和b的卷积就是$ ( a_0b_0, a_1b_0+a_0b_1, a_2b_0+a_1b_1+a_0b_2, a_1b_2+a_2b_1, a_2b_2 ) \(
    即可以看作两个多项式\)A(x)=a_0+a_1x1+a_2x2和B(x)=b_0+b_1x+b_2x^2\(相乘。
    所以卷积就可看做多项式乘法。那些形如\)c_k = \Sigma_{i=0}^{k}a_ib_{k-i}$卷积的就可以类比成多项式乘法 ,就可通过fft快速求解。

  2. 多项式

    • 多项式有两种表达形式
    • 一种是系数表达形如\(A(x) = \Sigma_{i=0}^{n-1} a_ix^i\)
    • 另一种是点值表达, 形如\((x_1, y_1), (x_2, y_2), (x_3, y_3)...(x_m, y_m)\), 且m>=n
    • 多项式的点值表示和插值表示可以互推
  3. \(w_n=e^{2\pi i/n}\)

    • \(w_n^{n/2+k}=-w_n^k\)
    • \(w_n^{2k}=w_{n/2}^k\)
    • $ \Sigma_{j=0}{n-1}w_n{jk}= (k%n==0)?n:0$

傅里叶变换的原理

在傅里叶变换里面系数表达和点值表达如何互推?

一个暴力点的方法

系数表达->点值表达 \(y_k = \Sigma_{i=0}^{n-1}a_iw_n^{ki}\)

点值表达->系数表达 \(a_k = 1/n*\Sigma_{i=0}^{n-1}y_iw_n^{-ki}\)

证明看黑板

所以不论是系数表达推点值表达还是点值表达推系数表达方法都一样

如何快速的求出上面的式子呢

假设 $A(x)=a_0+a_1x+a_2x2+...+a_{n-1}x{n-1} \(
那\)A(x)=(a_0+a_2x2+a_4x4+...+a_{n-2}x{n-2})+(a_1x+a_3x3+..+a_{n-1}x^{n-1})\(
令\)A_0(x)=a_0+a_2x+a_4x2+...+a_{n-2}x{n/2-1}, A_1(x)=a_1+a_3x+a_5x+...+a_{n-1}x^{n/2-1}\(
则\)A(x) = A_0(x2)+xA_1(x2)\(
那么就可以用\)A_0(w_{n/2}^0), A_0(w_{n/2}^1) ... A_0(w_{n/2}^{n/2-1})$ 和 \(A_1(w_{n/2}^0), A_1(w_{n/2}^1) ... A_1(w_{n/2}^{n/2-1}\)

得出\(A(w_{n}^0), A(w_{n}^1) ... A(w_{n}^{n-1})\)

\(所以可以采用递归分治的办法求解FFT\)

分治图

void ntt(ll* a, int n, int t) {
rep(i, 0, n) {
int rv = rev(i,n);
if(i<rv) swap(a[i], a[rv]);
}
ll g=1;
if(t==1) g=3;
else g=inv[3];
for(int m=2; m<n+1; m<<=1) {
ll wm= mypow(g, (MOD-1)/m);
int mid=m>>1;
for(ll *p=a; p<a+n; p+=m) {
ll w=1;
rep(i, 0, mid) {
ll t=w*(p[mid+i]);
p[mid+i] = p[i]-t;
p[i] = p[i]+t;
w = w*wm%MOD; p[i]=%MOD, p[mid+i]%=MOD;
}
}
}
if(t==-1) {
rep(i, 0, tn)
a[i]=a[i]*inv[n]%MOD;
}
}
void fft(ll *a, ll* b, int n) {
int tn=MAXN;
while(tn/4>=n) tn>>=1;
ntt(a, tn, 1);
ntt(b, tn, 1);
rep(i, 0, tn)
a[i] = a[i]*b[i]%MOD;
ntt(a, tn, -1);
}

NTT \(w_n=g^{(P-1)/n}\%P\)

FFT 专题讲解的更多相关文章

  1. Activity Process Task Application 专题讲解

    Activity Process Task Application 专题讲解 Activity.和进程 为了阅读方便,将文档转成pdf http://files.cnblogs.com/franksu ...

  2. 13Mybatis_SqlMapConfig.xml专题讲解

    Mybatis的SqlMapConfig.xml中以下的标签: properties(属性) settings(全局配置参数) typeAliases(类型别名) typeHandlers(类型处理器 ...

  3. Android中View的绘制流程(专题讲解)

    Android中的UI视图有两种方式实现:.xml文件(实现代码和UI的分离)和代码实现. Android的UI框架基本概念: 1. Activity:基本的页面单元,Activity包含一个Wind ...

  4. LeetCode递归 -2(Recursion) 培训专题 讲解文章翻译 (附链接) (2019-04-09 15:50)

    递归 - 空间复杂度  在本文中, 我们将讨论如何分析递归算法的空间复杂度. 在计算递归算法的空间复杂度时,最需要考虑的两个部分就是: 递归相关空间 (recursion related space) ...

  5. LeetCode 递归(Recursion) 培训专题 讲解文章翻译 (附链接)

     递归 - 时间复杂度 在本文中, 我们主要介绍如何分析递归算法程序中的时间复杂度.. 在一个递归程序中, 它的时间复杂度 O(T) 一般来说就是他总共递归调用的次数 (定义为 R) 以及每次调用时所 ...

  6. 火柴排队(NOIP2013)(附树状数组专题讲解(其实只是粗略。。。))

    原题传送门 首先,这道题目是一道神奇的题. 看到这道题,第一眼就觉得2个数组排个序,然后一一对应的时候一定差值最小. 由于我们可以将这2个数列同时进行调换. 所以我们先把2个数列排个序. 第二个序列中 ...

  7. H3C路由器登录策略专题讲解

    password-control login-attempt login-times [ exceed { lock | lock-time time | unlock } ] undo passwo ...

  8. javascript进阶系列专题:作用域与作用域链

    字面意思,作用域是指变量和函数的作用范围,换言之,作用域决定了变量和函数的可见性和有效时间.javascript作用域是用函数来区分,与其他语言的大括号不同. for (var i=0; i<5 ...

  9. 在 Visual Studio 2013 中创建 ASP.NET Web 项目(0):专题导航 [持续更新中]

    写在前面的话 随着 Visual Studio 2013 的正式推出,ASP.NET 和 Visual Studio Web 开发工具 也发布了各自的最新版本. 新版本在构建 One ASP.NET ...

随机推荐

  1. DOCKER 从入门到放弃(三)

    使用docker create [image-name] 创建一个容器 创建一个nginx镜像的容器,由于没有指定各项参数,容器实用默认参数,创建后并不会启动,并将容器的ID输出到终端,如果本地没有镜 ...

  2. vijos1698题解

    题目: 船体的结构是不能随意修改的..那样会破坏整艘船和谐的韵律.. 虽然说.如果沿岸航行的话是不会预见太大的海浪的..但是还要小心保护轨杆和船帆.. 毕竟对于小s这样的单轨帆船...轨杆和船帆如果受 ...

  3. JAVA 中数组的几种排序方法

    1.数组的冒泡排序   public void bubbleSort(int a[]) {      int n = a.length;      for (int i = 0; i < n - ...

  4. Unity3D文件读取

    Resources: 是作为一个Unity3D的保留文件夹出现的,也就是如果你新建的文件夹的名字叫Resources,那么里面的内容在打包时都会被无条件的打到发布包中.它的特点简单总结一下就是: 只读 ...

  5. 使用curl,libcurl访问Https

    编译curl,libcurl 下载curl源码(git clone https://github.com/curl/curl),在目录curl\winbuild\BUILD.WINDOWS.txt文件 ...

  6. Spring+SpringMVC+MyBatis深入学习及搭建(十三)——SpringMVC入门程序(二)

    1.非注解的处理器映射器和适配器 1.1非注解的处理器映射器 前面我们配置的org.springframework.web.servlet.handler.BeanNameUrlHandlerMapp ...

  7. C++第四篇--重载_指针_引用

    C++第四篇--重载_指针_引用 1. 基础知识 重载:函数名相同,根据参数不同(类型.数量.顺序不同)调用同名函数 指针和引用:引用就是别名,引用时必须初始化,引用你定义的变量. int a; in ...

  8. 浅谈JVM与内存分配

    一.程序内存分配 初始内存分配 当一个程序准备运行时,它首先向java虚拟机要内存,但是java虚拟机本身没有权限,它只能向操作系统申请内存,此时java虚拟机会拥有一个初始内存, 此处额外说明一下e ...

  9. Java多线程缓存器简单实现

    package com.charles.utils; import java.util.HashMap; import java.util.Map; import java.util.concurre ...

  10. 使用Dubbo、JSF等RPC框架时,对于异常的处理

    无论是Dubbo还是JSF等RPC框架,一般都会把接口分为2部分: 1,服务端(provider) 2,客户端(consumer) 由于,客户端与服务端可能不在同一个应用中,所以客户端一般在调用服务端 ...