快速傅里叶变换FFT(Fast Fourier Transform)

快速傅里叶变换(Fast Fourier Transform,FFT)用来计算离散傅里叶变换(Discrete Fourier Transform,DFT)及其逆变换(IDFT),本质上多项式乘法实际上是多项式系数向量的卷积(convolution)
(以上百度)

看了很多网上关于FFT的讲解,有一些是直接忽略了公式的推导,另外一些有推导,但是推导中的细节却没有讲清楚。本着不懂就学的心态,我把FFT的思维和推导细节用公式讲清楚,方便后人能更细致地学习FFT。


在了解FFT之前,需要有一些前置的知识,以下为目录。

  1. 欧拉函数
  2. 复数及单位根
  3. 多项式的系数、点值表达和多项式相乘
  4. FFT的思维过程(Cooley-Tukey 算法)
    4.1 DFT
    4.2 IDFT
  5. FFT的实现过程以及FNT

一、欧拉函数

\tag{1} e^{i\theta}=cos\theta+isin\theta其中i为虚数单位即\tag{2} i^2 = -1即虚数单位\sqrt{-1}

二、复数及单位根

复数形式:a+bi,其中i为虚数单位


复数乘法:对于两个复数A=a_0+b_0iB=a_1+b_1i,则A*B = a_0a_1+b_0b_1i^2+(a_0b_1+a_1b_0)i=a_0a_1-b_0b_1+(a_0b_1+a_1b_0)i
由于欧拉公式(见公式1)令a=rcos\theta , b=rsin\theta则复数\tag{3} a+bi = r(cos\theta + isin\theta)=re^{i\theta}
其中r为该复数所在复平面圆的半径,\theta为该复数在复平面中的幅角。则两个复数为A=r_Ae^{i\theta_{A}}, B=r_Be^{i\theta_{B}},即\tag{4}A*B=r_Ar_Be^{i(\theta_{A}+\theta_{B})}
根据4式可得,两个复数的相乘可以看作是幅角相加,模长相乘


单位根:对于满足w^n=1方程的复数,我们称其为n次单位根。由于根据复数乘法,我们可知:复数相乘为幅角相加,模长相乘。则对于每个单位根,模长为1,幅角的n倍为0。即r=1,sin{(n \theta )} = 0(易得)。

为了保证幅角的n倍始终为0,由于sin(2k\pi)=0,k=0,1,2,...,n这个性质,我们可以将单位根表示为e^{\frac{2k\pi i}{n}},k=0,1,2,...,n,其中r=1,\theta=\frac{2k\pi }{n},k=0,1,2...,n

我们发现无论k取值,\theta的n倍始终为0。

w_n=e^{\frac{2k\pi}{n}},则n次单位根可以表示为w_n^{0},w_n^{1},w_n^{2},...,w_n^{n-1}

三、多项式的系数、点值表达和多项式相乘

多项式的系数表达:给定一个多项式\tag{5} f(x)=\sum_{i=0}^{n}a_ix^i =a_0x^0+a_1x^1+a_2x^2+...+a_{n-1}x^{n-1}+a_nx^n
其中\mathbf {a} = (a_0,a_1,a_2,...,a_n)为系数向量


多项式的点值表达:给定一个多项式如公式(5),我们将其离散化,设取n+1(这里为什么是n+1项,将在第四节中讲到)互不相同的值P = \lbrace p_0,p_1,p_2,p_3,...,p_{n-1},p_n\rbrace,将其代入可得\tag{6} A(x)=\lbrace(p_0,f(p_0)),(p_1,f(p_1)),(p_2,f(p_2)),...,(p_n,f(p_n))\rbrace,则A(x)f(x)P上的点值表达。


多项式系数表达的乘法:给定两个多项式\tag{6} A(x)=\sum_{i=0}^na_ix^i=a_0x^0+a_1x^1+a_2x^2+...+a_{n-1}x^{n-1}+a_nx^n
\tag{7} B(x)=\sum_{i=0}^nb_ix^i=b_0x^0+b_1x^1+b_2x^2+...+b_{n-1}x^{n-1}+b_nx^n
则多项式系数表达的乘法为\tag{8} C(x)=A(x)*B(x)=\sum_{i=0}^{2n}c_ix^i
其中:
\tag{9} c_i = \sum_{j+k=i,0\leq j,k\leq n} a_jb_kx^i
有式(9)可得,计算复杂度为\omicron(n^2)


多项式点值表达的乘法:给定两个多项式如式(6)与(7),则其在P=\lbrace p_0,p_1,p_2,...,p_{n-1},p_n \rbrace上的点值表达分别为:
\tag{10} A'(x)=\lbrace(p_0,A(p_0)),(p_1,A(p_1)),(p_2,A(p_2)),...,(p_n,A(p_n))\rbrace
\tag{11} B'(x)=\lbrace(p_0,B(p_0)),(p_1,B(p_1)),(p_2,B(p_2)),...,(p_n,B(p_n))\rbrace
则多项式点值表达的乘法为
\tag{12} C'(x) =A'(x)*B'(x)=\lbrace (p_0,A(p_0)*B(p_0)),(p_1,A(p_1)*B(p_1)),(p_2,A(p_2)*B(p_2)),...,(p_n,A(p_n)*B(p_n))\rbrace
可见,当我们已知A'(x)和B'(x)即可在\omicron(n)的复杂度下求得结果多项式的点值表达。

四、FFT的思维过程(Cooley-Tukey 算法)

对于一个多项式的乘法,根据上述前置知识的补充,我们可以得知:降低多项式乘法复杂度的方法是将常见的多项式系数表达转变为多项式的点值表达,做完点值表达的乘法后,最后再将点值表达转化为系数表达,即可完成多项式乘法。
所以问题转变为:
1.如何将多项式系数表达转变为多项式点值表达
2.如何将多项式点值表达转变为多项式系数表达
由此引出了离散傅里叶变换 DFT(Discrete Fourier Transformation)和逆离散傅里叶变换 IDFT(Inverse Discrete Fourier Transformation)

4.1 离散傅里叶变换DFT(Discrete Fourier Transformation)

离散化多项式的一种方法是将值代入到多项式中,依次求出点值。显而易见,这种方法的复杂度是\omicron(n^2)的,这与我们降低复杂度的想法是冲突的。
于是我们引入了FFT的经典算法——Cooley-Tukey 算法,来降低离散化的复杂度,这是一个基于分治策略的算法。


给定一个多项式\tag{13} A(x)=\sum_{i=0}^na_ix^i=a_0x^0+a_1x^1+a_2x^2+...+a_{n-1}x^{n-1}+a_nx^n
我们将其根据奇偶项分成两个项数相同的多项式(将多项式补充到2^k项,n\leq 2^k,补充项数系数为0。PS:为什么是2^k项呢,后续将会提及):
\tag{14} A_0(x)=\sum_{i=0}^{\frac{n}{2}-1}a_{2i}x^{2i}=a_0x^0+a_2x^2+...+a_nx^n
\tag{15} A_1(x)=\sum_{i=0}^{\frac{n}{2}-1}a_{2i+1}x^{2i+1}=x\sum_{i=0}^{\frac{n}{2}-1}a_{2i+1}x^{2i}=a_1x^1+a_3x^3+...+a_{n-1}x^{n-1}
显而易见:A(x)=A_0(x)+A_1(x)


在进一步之前:我们需要返回单位根的知识点。根据n次单位根的表达w_n^k,k=0,1,2,...,n-1,我们可以获得一个等式
\tag{16} w_n^{\frac{n}{2}+k}=w_n^{\frac{n}{2}}·w_n^k=-w_n^k


我们将其代入式子(14),(15)可得:
\tag{17} A_0(w_n^k)=\sum_{i=0}^{\frac{n}{2}-1}a_{2i}w_n^{2ki}=\sum_{i=0}^{\frac{n}{2}-1}a_{2i}w_{\frac{n}{2}}^{ki}
\tag{18} A_1(w_n^k)=\sum_{i=0}^{\frac{n}{2}-1}a_{2i+1}w_n^{2ki+1}=w_n^k\sum_{i=0}^{\frac{n}{2}-1}a_{2i+1}w_n^{2ki}=w_n^k\sum_{i=0}^{\frac{n}{2}-1}a_{2i+1}w_{\frac{n}{2}}^{ki}
\tag{19} A(w_n^k)=\sum_{i=0}^{\frac{n}{2}-1}a_{2i}w_{\frac{n}{2}}^{ki}+w_n^k\sum_{i=0}^{\frac{n}{2}-1}a_{2i+1}w_{\frac{n}{2}}^{ki}
至此我们发现原本需要n次代入值的等式降低到了\frac{n}{2}次,依次递归下去,则我们只需要递归log_2{n}次即可在\omicron(1)的复杂度下求得式子,即我们求得n+1个点值对的复杂度为\omicron(nlog_2n),是可以接受的复杂度。


为了更加严谨的证明,以下过程供还有疑问的读者参考
由于式子(16)可得w_n^k=-w_n^{\frac{n}{2}+k}
\tag{20} A(w_n^{\frac{n}{2}+k})=\sum_{i=0}^{\frac{n}{2}-1}a_{2i}w_{\frac{n}{2}}^{(\frac{n}{2}+k)i}+w_n^{\frac{n}{2}+k}\sum_{i=0}^{\frac{n}{2}-1}a_{2i+1}w_{\frac{n}{2}}^{(\frac{n}{2}+k)i}=\sum_{i=0}^{\frac{n}{2}-1}a_{2i}w_{\frac{n}{2}}^{ki}-w_n^k\sum_{i=0}^{\frac{n}{2}-1}a_{2i+1}w_{\frac{n}{2}}^{ki}
其中求和中的w_{\frac{n}{2}}^{(\frac{n}{2}+k)i}直接被替换为w_{\frac{n}{2}}^{ki}的原因是,经过平方以后,负号被抵消。
复杂度公式则为T(n)=T(\frac{n}{2})+\omicron({n})=\omicron({nlog_2n})


以上为Cooley-Tukey离散傅里叶变换DFT的思路。

我们回到为什么我们要将多项式补充到2^k项的问题上,我们发现,根据Cooley-Tukey 分治的策略,我们需要每次分治的两部分都有同样的项数,则总项数必须为2^k项。

4.2 逆离散傅里叶变换IDFT(Inverse Discrete Fourier Transformation)

经过DFT,我们将多项式的系数表达转换为多项式的点值表达。在完成乘法运算以后,我们为了获取系数的变换,需要将多项式的点值表达转换为多项式的系数表达。这时我们使用的方法是逆离散傅里叶变换IDFT,他是DFT的逆。


求解IDFT的过程实际上是一个求解线性方程的问题,给出n个线性方程为:
\tag{21}\begin{cases} a^0(w_n^0)^0+a^1(w_n^0)^1 + ... +a^{n-1}(w_n^0)^{n-1} = A(w_n^0)\\ a^0(w_n^1)^0+a^1(w_n^1)^1 + ... +a^{n-1}(w_n^1)^{n-1} = A(w_n^1)\\...\\a^0(w_n^{n-1})^0+a^1(w_n^{n-1})^1 + ... +a^{n-1}(w_n^{n-1})^{n-1} = A(w_n^{n-1}) \end{cases}矩阵形式如下:
\tag{22} \begin{bmatrix} (w_n^0)^0 & (w_n^0)^1&...&(w_n^0)^{n-1} \\ (w_n^1)^0 & (w_n^1)^1&...&(w_n^1)^{n-1}\\...\\(w_n^{n-1})^0 & (w_n^{n-1})^1&...&(w_n^{n-1})^{n-1} \end{bmatrix} \begin{bmatrix}a^0\\a^1\\...\\a^{n-1}\end{bmatrix}= \begin{bmatrix}A(w_n^0)\\A(w_n^1)\\...\\A(w_n^{n-1})\end{bmatrix}
假设上述矩阵为\mathbf V,则对于矩阵\mathbf D中值d_{ij}=-v_n^{ij}
\tag{23} D=\begin{bmatrix} (w_n^{-0})^0 & (w_n^{-0})^1&...&(w_n^{-0})^{n-1} \\ (w_n^{-1})^0 & (w_n^{-1})^1&...&(w_n^{-1})^{n-1}\\...\\(w_n^{-(n-1)})^0 & (w_n^{-(n-1)})^1&...&(w_n^{-(n-1)})^{n-1} \end{bmatrix}
设两个矩阵相乘以后的结果为\mathbf E=\mathbf D ·\mathbf V
\tag{24} e_{ij} =\sum_{k=0}^{n-1}d_{ik}v_{kj}=\sum_{k=0}^{n-1}w_n^{-ik}w_n^{kj}=\sum_{k=0}^{n-1}w_n^{k(j-i)}
j=i时,e_{ij}=n
j\neq i时,e_{ij}=\frac{1-(w_n^{j-i})^n}{1-(w_n^{j-i})}=0
(其中由于w_n^{j-i}n次单位根,又因为n次单位根的n次为1,所以上式成立)
所以\mathbf E=n\mathbf I,\mathbf D=\frac{1}{n}\mathbf V^{-1}(其中\mathbf I 为单位矩阵)
\tag{25} \begin{bmatrix}a^0\\a^1\\...\\a^{n-1}\end{bmatrix}= \frac{1}{n}\begin{bmatrix} (w_n^{-0})^0 & (w_n^{-0})^1&...&(w_n^{-0})^{n-1} \\ (w_n^{-1})^0 & (w_n^{-1})^1&...&(w_n^{-1})^{n-1}\\...\\(w_n^{-(n-1)})^0 & (w_n^{-(n-1)})^1&...&(w_n^{-(n-1)})^{n-1} \end{bmatrix}\begin{bmatrix}A(w_n^0)\\A(w_n^1)\\...\\A(w_n^{n-1})\end{bmatrix}
则IDFT便是将w_n^{i}=w_n^{-i}对结果再做一次DFT,即可获得最后的系数。

回到为什么要有n+1取值的原因在于:为了多项式的点值表达转换为多项式的系数表达,我们需要系数矩阵可逆,为了使得系数矩阵可逆,需要满足矩阵满秩,为了满足矩阵满秩,我们需要使得矩阵的行数大于等于列数。则对于有n+1个系数的多项式,必须要有n+1个以上的取值

四、FFT的实现过程以及FNT

在具体实现FFT的过程中,还需要考虑到对于每一次递归我们如何进行合理的划分。于是这里引入bitreverse算法,也叫做蝴蝶变换。

16个数进行DFT变换划分过程

如图所示,这是16个数进行DFT变换划分的过程。网上的步骤说的很复杂,简单来说,对于DFT的第i次划分,二进制第i位上为1的分为一类,为0的分为一类,即可完成。
举例说明:

  1. 对于步骤一来说,第1次划分,二进制第1位(即右边第一位)为0的分到左边,为1的分到右边。我们可以发现左边都为偶数,右边都为奇数。(因为奇数二进制第1位为1)
  2. 对于步骤二的左半部分,第2次划分,二进制第2位(即右边第二位)为0的分到左边,为1的分到右边。同理,右半部分,第2次划分,也是二进制第2位(即右边第二位)为0的分到左边,为1的分到右边。
  3. 同理,对每次的分组进行划分,即可将所有的数字划分到对应的组中。

通过这种划分方法,我们同时还能总结出另外一个规律,对于对于2^k个数字中的任意一个位置的数字,假设原本位置为x,二进制反转的函数为inverse(x),则最后其所在的位置为inverse(x)(第一个位置为0),其中xk位二进制。
举例说明:
对于16(2^4)个数字中的14(1110),则其4位二进制的反转为0111(7),则其最后的位置为第7位(ps:图中没有继续将所有的数字划分到每组一个,读者可自行划分检验)


这里可以补充一个写法:假如我们将原数组定义为A,经过反转后的数组定义为B,则A[reverse(i)]=B[i]
又因为如果i是偶数,则i=(i>>1)<<1,则对于reverse(i)=reverse(i>>1)>>1,但考虑到如果是i是奇数,则((i>>1) << 1) | 1,则对于reverse(i)=reverse(i>>1)>>1|(1<<(k-1))其中k为满足2^k>n的最小值。
综合写可以写成reverse(i)=reverse(i>>1)>>1 | (i\&1) << (k-1)
通过这个写法,我们可以直接写出所有数字经过DFT划分后的结果。

参考:从多项式乘法到快速傅里叶

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容

  • 本章涉及知识点:1、多项式乘法的时间复杂度2、多项式的表示:系数3、多项式的表示:点值4、复数的表示5、单位复数根...
    PrivateEye_zzy阅读 9,593评论 0 11
  • 概述   希尔伯特空间是一个完备的内积空间,其标准正交函数系,直观来看就是向量空间中基的延伸。其为基于任意正交系上...
    殉道者之花火阅读 1,722评论 1 5
  • 快速傅里叶变换(Fast Fourier Transform,FFT)是一种可在时间内完成的离散傅里叶变换(Dis...
    madeirak阅读 3,529评论 0 4
  • 前言   本文是个人学习心得的分享,希望大家在阅读文章后能在评论中一起学习交流!另外还可以访问我的HelloCUD...
    liadrinz阅读 7,366评论 0 3
  • 【注】参考自算法导论。 1. 简介 快速傅里叶变换(FFT)是实现离散傅里叶变换(DFT)和离散傅里叶逆变换(ID...
    BlueHeart0621阅读 425评论 0 1