问题:

  已知$A=a_{0..n-1}$, $B=b_{0..n-1}$, 求$C=c_{0..2n-2}$,使:

$$c_i = \sum_{j=0}^ia_jb_{i-j}$$

  定义$C$是$A$,$B$的卷积,记作

$$C = A * B$$

  例如多项式乘法等。

  朴素做法是按照定义枚举$i$和$j$,但这样时间复杂度是$O(n^2)$.

  能不能使时间复杂度降下来呢?

点值表示法:

  我们把$A$,$B$,$C$看作多项式。

  即:

$$A(x) = \sum_{i=0}^{n-1}a_ix^i$$

  将$A=\left\{(x_1,A(x_1)), (x_2,A(x_2)), (x_3,A(x_3))...\right\}$叫做$A$的点值表示法。

  由于$A$是$n-1$次多项式,我们恰好需要$n$个点值来确定它。

  那么使用点值表示法做多项式乘法就很简单了:对应项相乘(这样的话要将$n$扩大一倍,因为$C$的次数约为$2n$)。

  那么,如何将$A$和$B$转换成点值表示法,再将$C$转化回系数表示法(即最初的表示方法)呢?

  如果任取$n$个点,按照定义计算,那么还是$O(n^2)$的。

  这样就要用到快速傅里叶变换。

快速傅里叶变换:

  既然任取$n$个点,按照定义计算太慢,就要找一些特殊点。

  我们用$n$个$n$次单位复数根($1$的$n$次方根,涉及到复数,$1$的方根不止$1$和$-1$)来计算:

  根据欧拉公式,$e^{i\theta}=\cos\theta+i\sin\theta$(其中$i$是虚数单位),那么$e^{2\pi i}=\cos(2\pi)+i\sin(2\pi)=1$.

  所以1的n次方根是$\omega_n^k=e^{\frac{2k\pi i}n}\qquad0\leq k<n$。

  其中$\omega_n^1 = e^{\frac{2\pi i}n}$是主$n$次单位根,那么所有$n$次单位复数根都是它的幂。

  我们要求出$A(\omega_n^k)$,就要采用分治思想。

  我们将奇偶系数分离(先假设n为偶数),即定义

$$A_0(x)=a_0 + a_2* x + a_4 * x^2 +\cdots=\sum_{i=0}^{\frac{n}2-1}a_{2i}x^i$$

$$A_1(x)=a_1 + a_3* x + a_5 * x^2 +\cdots=\sum_{i=0}^{\frac{n}2-1}a_{2i+1}x^i$$

  那么$A(x)=A_0(x^2) + xA_1(x^2)$。

  要计算$A(\omega_n^k)=A_0\left[(\omega_n^k)^2\right] + \omega_n^kA_1\left[(\omega_n^k)^2\right]$,

  就要用到$(\omega_n^k)^2 = \omega_{n/2}^{k\,mod (n/2)}$(证略)。

  所以$A(\omega_n^k)=A_0\left(\omega_{n/2}^{k\,mod (n/2)}\right) + \omega_n^kA_1\left(\omega_{n/2}^{k\,mod (n/2)}\right)$

  我们发现$A_0$,$A_1$都是$n/2$项的,且只需要算$\omega_{n/2}^k$的值,那么这就和开始的问题一样了,可以分治。

  边界也很容易:$n=1$的时候$A_0$本身就是值。

  合并解。

$$A(\omega_n^k)=A_0\left(\omega_{n/2}^{k\,mod (n/2)}\right) + \omega_n^kA_1\left(\omega_{n/2}^{k\,mod (n/2)}\right)$$

  那么可以$A(\omega_n^k), A(\omega_n^{k+n/2})$一起算$(0\leq k<n/2)$ :

    令$u = A_0(\omega_{n/2}^k), t = \omega_n^kA_1(\omega_{n/2}^k)$,

    那么

$$A(\omega_n^k)=u + t$$

$$\begin{aligned}& \quad A(\omega_n^{k+n/2}) \\
&= A_0(\omega_{n/2}^k) + \omega_n^{k+n/2}A_1(\omega_{n/2}^k)\\
&= A_0(\omega_{n/2}^k) + \omega_n^k\omega_n^{n/2}A_1(\omega_{n/2}^k)\\
&= A_0(\omega_{n/2}^k) - \omega_n^kA_1(\omega_{n/2}^k)\\
&= u-t\end{aligned}$$

  所以这样就能算出$A$的点值表示法。

  一个问题:分治要求$n$是$2$的幂,不是怎么办? 补$0$, 直到$n$是$2$的幂。

  时间复杂度:

  $$T(n)=2T(n/2)+O(n)$$

  直接观察或者应用主定理都可得出$T(n)=O(nlogn)$

  剩下的问题:如何把C转化回系数表示法。

逆变换:

  我们把C做一遍快速傅立叶变换,只是求的是$\omega_n^n, \omega_n^{n-1}, \cdot,\omega_n^1$的值而不是$\omega_n^0, \omega_n^1, \cdot,\omega_n^{n-1}$的值,最后每一项除以n即可。

  证明(实际上可以利用逆矩阵,但我就写的麻烦一点吧qwq):

这样我实际上是求了$c_i=\frac1n\sum_{k=0}^{n-1}(\omega_n^{-i})^kC(\omega_n^k)$。我们来直接证明这是正确的。

$$\begin{aligned}
\frac1n\sum_{k=0}^{n-1}(\omega_n^{-i})^kC(\omega_n^k)&=\frac1n\sum_{k=0}^{n-1}\omega_n^{-ki}\sum_{j=0}^{n-1}c_j(\omega_n^k)^j\\
&=\frac1n\sum_{j=0}^{n-1}c_j\sum_{k=0}^{n-1}\omega_n^{jk}\omega_n^{-ki}\\
&=\frac1n\sum_{j=0}^{n-1}c_j\sum_{k=0}^{n-1}(\omega_n^{j-i})^k
\end{aligned}$$

$\sum_{k=0}^{n-1}(\omega_n^{j-i})^k$是一个公比为$\omega_n^{j-i}$的等差数列。在$i=j$时$n$项都为$1$,显然其值为$n$;$j\neq i$的时候根据等差数列求和公式他就等于

$$\frac{1-(\omega_n^{j-i})^n}{1-\omega_n^{j-i}}$$

而$(\omega_n^{j-i})^n$等于$1$,所以上式就是$0$。于是
$$\frac1n\sum_{k=0}^{n-1}(\omega_n^{-i})^kC(\omega_n^k)=\frac1n\sum_{j=0}^{n-1}c_j\sum_{k=0}^{n-1}(\omega_n^{j-i})^k=\frac1n\sum_{j=0}^{n-1}c_j\times n[i == j]=c_i$$
证毕。

  1. #include <algorithm>
  2. #include <cmath>
  3. const double pi = acos(-1.0);
  4. struct complex{
  5. double real, impl;
  6. complex(double r = 0.0, double i = 0.0) : real(r), impl(i) {}
  7. friend complex operator+(const complex &a, const complex &b) {
  8. return complex(a.real + b.real, a.impl + b.impl);
  9. }
  10. friend complex operator-(const complex &a, const complex &b) {
  11. return complex(a.real - b.real, a.impl - b.impl);
  12. }
  13. friend complex operator*(const complex &a, const complex &b) {
  14. return complex(a.real * b.real - a.impl * b.impl, a.impl * b.real + b.impl * a.real);
  15. }
  16. friend complex operator/(const complex &a, double b) {
  17. return complex(a.real / b, a.impl / b);
  18. }
  19. };
  20. using std::swap;
  21. void FFT(complex* P, int len, int opt) {
  22. for (int i = , j = , k; i < len; ++i) {
  23. for (k = len >> ; j & k; k >>= ) j ^= k;
  24. j ^= k;
  25. if (i < j) swap(P[i], P[j]);
  26. }
  27. for (int h = ; h <= len; h <<= ) {
  28. complex wn = complex(cos(opt * * pi / h), sin(opt * * pi / h));
  29. for (int j = ; j < len; j += h) {
  30. complex w = complex(1.0, .);
  31. for (int t = ; t < h / ; ++t, w = w * wn) {
  32. complex tmp1 = P[t + j], tmp2 = P[t + j + h / ];
  33. P[t + j] = tmp1 + tmp2 * w;
  34. P[t + j + h / ] = tmp1 - tmp2 * w;
  35. }
  36. }
  37. }
  38. if (opt == -)
  39. for (int i = ; i < len; ++i)
  40. P[i] = P[i] / len;
  41. }

FFT

Fast Fourier Transform ——快速傅里叶变换的更多相关文章

  1. 【OI向】快速傅里叶变换(Fast Fourier Transform)

    [OI向]快速傅里叶变换(Fast Fourier Transform) FFT的作用 ​ 在学习一项算法之前,我们总该关心这个算法究竟是为了干什么. ​ (以下应用只针对OI) ​ 一句话:求多项式 ...

  2. 数字图像处理实验(5):PROJECT 04-01 [Multiple Uses],Two-Dimensional Fast Fourier Transform 标签: 图像处理MATLAB数字图像处理

    实验要求: Objective: To further understand the well-known algorithm Fast Fourier Transform (FFT) and ver ...

  3. 「学习笔记」Fast Fourier Transform

    前言 快速傅里叶变换(\(\text{Fast Fourier Transform,FFT}\) )是一种能在\(O(n \log n)\)的时间内完成多项式乘法的算法,在\(OI\)中的应用很多,是 ...

  4. $\mathcal{FFT}$·$\mathcal{Fast \ \ Fourier \ \ Transformation}$快速傅立叶变换

    \(2019.2.18upd:\) \(LINK\) 之前写的比较适合未接触FFT的人阅读--但是有几个地方出了错,大家可以找一下233 啊-本来觉得这是个比较良心的算法没想到这么抽搐这个算法真是将一 ...

  5. Fast Walsh-Hadamard Transform——快速沃尔什变换

    模板题: 给定$n = 2^k$和两个序列$A_{0..n-1}$, $B_{0..n-1}$,求 $$C_i = \sum_{j \oplus k = i} A_j B_k$$ 其中$\oplus$ ...

  6. Fast Walsh-Hadamard Transform——快速沃尔什变换(二)

    上次的博客有点模糊的说...我把思路和算法实现说一说吧... 思路 关于快速沃尔什变换,为了方便起见,我们采用线性变换(非线性变换不会搞). 那么,就会有一个变化前各数值在变换后各处的系数,即前一篇博 ...

  7. 快速傅里叶变换(Fast Fourier Transform, FFT)和短时傅里叶变换(short-time Fourier transform,STFT )【资料整理】【自用】

    1. 官方形象展示FFT:https://www.bilibili.com/video/av19141078/?spm_id_from=333.788.b_636f6d6d656e74.6 2. 讲解 ...

  8. 1250 Super Fast Fourier Transform(湘潭邀请赛 暴力 思维)

    湘潭邀请赛的一题,名字叫"超级FFT"最终暴力就行,还是思维不够灵活,要吸取教训. 由于每组数据总量只有1e5这个级别,和不超过1e6,故先预处理再暴力即可. #include&l ...

  9. Fast Fourier Transform

    写在前面的.. 感觉自己是应该学点新东西了.. 所以就挖个大坑,去学FFT了.. FFT是个啥? 挖个大坑,以后再补.. 推荐去看黑书<算法导论>,讲的很详细 例题选讲 1.UOJ #34 ...

随机推荐

  1. new表达式如何创建对象

    new表达式如何创建对象 前言 刚学java时曾一度认为,构造器中this指向是当前类型的对象,当我们调用new表达式时,由父类的构造器生成对象的一部分并初始化,然后再由子类的构造器补充成完整的对象并 ...

  2. java如何停止一个运行的线程?

    关于线程的一点心得 //首先导入需要的包 improt java.util.Timer;import java.io.File;import java.util.TimerTask; //首先需要创建 ...

  3. 9.Linux系统引导流程

    一.Linux系统引导流程 当我们按下主机电源键的那时候开始,主板上的CMOS/BIOS模块将进行固件自检,以此检查各个硬件是否正确连接. 在Linux引导流程中,一般可以分为以下几个主要过程: 1. ...

  4. Java中log4j的使用

    前言 距离上一篇文章又过去好长时间了,这段时间一直忙于工作,已经从net彻底转向Java了.工作也慢慢的步入正轨了,自己独自完成了一个小项目,不过工作中遇到了一些问题,还是得到了同学和同事的帮助.本来 ...

  5. HNOI2017前被虐记及感悟

    本文所记录的时间以HNOI2017第一天考试时间为DAY1,前一天为DAY0,以此类推. 本文记载了博主从HNOI2017开始前一周进行全真模拟考试的被虐过程和结果.文章内可能包含博主的不良情绪,如果 ...

  6. js 模板引擎

    template = document.querySelector('#template').innerHTML, result = document.querySelector('.result') ...

  7. 利用 force recovery 解决服务器 crash 导致 MySQL 重启失败的问题

    小明同学在本机上安装了 MySQL 5.7.17 配合项目进行开发,并且已经有了一部分重要数据.某天小明在开发的时候,需要出去一趟就直接把电脑关掉了,没有让 MySQL 正常关闭,重启 MySQL 的 ...

  8. 函数指针|指针函数|C文件操作

    body,table { font-family: 微软雅黑; font-size: 10pt } table { border-collapse: collapse; border: solid g ...

  9. PROFINET有什么用

    “工业4.0”是当前制造业最热门的话题,所以不谈这个话题都不好意思跟同行们打招呼.“工业4.0”里面的一个重要内容是智慧工厂,工厂流水线设备之间通信,无论是传统的有线连接还是先进的无线连接与分布式控制 ...

  10. Linux安装redis及redis的php扩展。

    ------ redis安装,启动服务,开机启动,打开redis客户端 ------ yum install -y redis systemctl start redis systemctl enab ...