快速傅里叶变换(FFT)

2023-10-29

前言:

在学习FFT过程中看了很多博客,但发现在看博客的时候博客上的内容大多都晦涩难懂,于是乎想自己写一篇博客来记录一下自己学习的心得体会。

知其源:

先来讲讲FFT的起源:

快速傅里叶变换是1965年由J.W.库利和T.W.图基提出的。采用这种算法能使计算机计算离散傅里叶变换所需要的乘法次数大为减少,特别是被变换的抽样点数N越多,FFT算法计算量的节省就越显著。

它是根据离散傅氏变换的奇、偶、虚、实等特性,对离散傅立叶变换的算法进行改进获得的。

                                                                                                                                                                  (转载自百度百科)

举个例子:

                                                                       

上面的式子的时间复杂度为0(n^2),FFT能够把上式运算的时间复杂度将为O(nlogn)。

既然说到时间复杂度,那就来看看这里的O(logn),从O(n)到O(logn),显然用到了折半的思想,这里只是简单的提一下。

相信很多没有接触过这个类型的算法的人来说,对于怎么实现是一头雾水的,不积跬步无以至千里,下面就来聊聊FFT的前置知识。

1.1:多项式的系数表示法和点值表示法

为什么要了解这一部分的内容呢?

                                                        

上面是一个由系数为{a0,a1,a2.....an}组成的多项式。

在函数图像中,这个图像可以被(n+1)个点唯一表示,也就是说我们使用{(x0,y0).....(xn,yn))这n+1个点可以表示这个图像,如果不理解的话可以类比解方程,当一个方程里面有x个未知数的时候,就需要有x个方程,根据这个想法我们可以把上面的多项式使用矩阵乘积的形式进行表示:

                                

上面的一部分就是多项式的系数表示法,下面就来谈谈多项式的点值表示法.....

把这过多项式放到平面直角坐标系里面,然后把n+1个不同的点带入到这个多项式中就会得到n+1个不同的点(位置),那么这n+1个点就唯一确定了这个表达式,也就是说A(x)还可以表示成A(x)={(x0,A(x0)),(x1,A(x1)),......,(xn,A(xn))}。

1.2:线性卷积

在高精度乘法中,我们手推一下计算过程:

                                                                    

很显然,使用系数表示法进行高精度乘法的时间复杂度为0(n^2),我对这种乘法的理解是——这种乘法是横向进行的,369计算完之后计算246,最后计算123,但是呢现在如果转换一个方向的话就能改变时间复杂度,也就是我们把9看成一个部分(常数项),把66看成一个部分(x),把343看成一个部分(x^2)........那么时间复杂度就变成O(n)了。

                                                              

前面提到的多项式是从常数项开始到x^n的(低位到高位),而上面我举例的乘法运算是从高位到低位的,不过变化的只是方向由x^0到x^n变成了从x^n到x^0,想象一下,当我们把运算过程模拟上面的运算列出来后左下角有一个空白的三角形,长度分别为0、1、2、3.......如果你有编程基础的话,能够想象我们在打印图形的时候遇到这样的问题的解决方法,填空,或者说往后移,那么上面的公式就实现了后移操作。

 

你可别觉得前面提到的多项式的系数表示法和点值表示法没有用,这里可就派上大用场了~

你瞧,这里的把常数项,x,x^2......看做一个个整体不就是点系数表示法嘛。

这样我们就可以将多项式的系数表示法转换成点值表示法来做,那时间复杂度不就降下来了嘛!

不过多项式转换成点值表示的朴素算法是O(n2)的,很遗憾,我们白高兴了一场,现在既然不能把时间复杂度从O(n^2)降到O(n),那么我们就得想其它的办法把时间复杂度降下来,毕竟时间就是金钱呐!

于是乎,傅里叶"高呼":这个简单,我会。

1.3:离散傅里叶变换

1.3.1:复数

在R范围内sqrt(x)(x>=0),但是在复数范围内,就不是这样了,定义i^2=-1,一个复数z可以表示成z=a+bi(a、b∈R,其中a为实部,b为虚部,i为虚部单位)

sqrt(-1)=sqrt(1)*i(在复数集里面可以用i表示负数的平方根)

还可以把复述看成复平面直角坐标系上面的一个点: 

                                        

这个点(2,6)表示的复数就是2+6i,或者它表达的向量为(2,6),一个复数的共轭复数为虚部前面的数字*-1。

复数的运算如下:

复数不像点或者向量,它和实数一样可以进行四则运算

设两个负数分别为z1=a+bi,z2=c+di,那么:

                                                 

另外,复数相乘还有一个与极角有关的小性质(这里记为①):

                                             

C++里面的的STL提供了复数模板!

头文件:#include <complex>
定义: complex<double> x;

我的实现过程如下:

struct Complex{
    double x,y;//分别表示实部和虚部
    Complex(double _x=0.0,double _y=0.0){
        x=_x;
        y=_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);
    }
    Complex operator *(const Complex &b)const{
        return Complex(x*b.x-y*b.y,x*b.y+y*b.x);
    }
};

 

FFT就是将系数表示法转化成点值表示法相乘,再由点值表示法转化为系数表示法的过程,第一个过程叫做求值(DFT),第二个过程叫做插值(IDFT)

                                                                                                                                                            - Duan2baka

1.3.2:NFT

 傅里叶发明了一种办法:规定点值表示中的n个x为n个模长为1的复数(~~~~所以补充了复数的知识~~~~)。

对于模长为1的复数我是类比单位圆理解的~~~~

但是,傅里叶要用到的n个模长为1的复数可不是任意选择的,而是从一个圆中均匀选择的。

                                                         

我们记从(1,0)开始的n个点表示为,使用复数的形式表示为:,根据①我们可以知道就是的k次方。

带入到多项式中会得到一种特殊的点值表示,称为离散傅里叶变换!简单的说就是求值。

下面是一些性质及其证明:

性质一:

性质二:

性质三:

 

回到前面提到的系数表示法:

                                                

我们要将一个系数表示法的函数转换成点值表示法,那么我们就需要n+1组(不同的)数据带入到多项式里面,对于每组数据带入后的时间复杂度为O(n),那么总的时间复杂度就是O(n^2)。

这里我假设f(x)分为两部分,一个是奇数部分一个是偶数部分,表示如下:

根据前面提到的性质2我们可以再次进行化简:

在最开始我有提到折半的思想,也就是二分,想一下,上面式子里的两个部分不就可以使用二分来解决吗。

至于括号里面的,我们只需求出来,然后k次方就行了。

1.3.3:IDFT

最开始提到函数的矩阵表示,注意观察等号的两边。

我假设上面矩阵的三部分分别为A V C,那么最后的C就是我们最后要求的结果,而C是通过A和B求出来的,因为A和B位于等号的两端,所以我们可以通过C=V的逆矩阵*A进行表示。

一个矩阵乘上它的逆矩阵等于单位矩阵,根据这个性质我们可以求出逆矩阵

矩阵乘法是前一个矩阵的行中的元素乘上后一个矩阵中的列中的元素。

 

 

 

参考资料:

百度百科:https://baike.baidu.com/item/%E5%BF%AB%E9%80%9F%E5%82%85%E9%87%8C%E5%8F%B6%E5%8F%98%E6%8D%A2/214957?fromtitle=FFT&fromid=4766072&fr=aladdin

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

快速傅里叶变换(FFT) 的相关文章

  • numpy 中的 FFT 与 MATLAB 中的 FFT 没有相同的结果

    我有一个带有复数的向量 可以找到here https www dropbox com s ve0de4ebk41s8y2 data txt dl 1 无论是在 Python 中还是在 MATLAB 中 我正在计算ifft 转化为 ifft
  • 使用 Node.js 检测麦克风的音频

    我在 Raspberry Pi 上使用 node js 来控制一些仪器 我想让麦克风监听特定信号 例如 500 Hz 的音调 并在听到该信号时触发事件 查看了多个 node js 库后 node core audio https www n
  • 使用 JTransforms 库通过 FFT 计算自相关

    我正在尝试使用下面的代码计算时间序列中样本窗口的自相关性 我将 FFT 应用于该窗口 然后计算实部和虚部的幅度并将虚部设置为零 最后对其进行逆变换以获得自相关 DoubleFFT 1D fft new DoubleFFT 1D magCnt
  • 如何从信号中去除频率

    我想从信号中删除一个频率 一个峰值 并在没有它的情况下绘制我的函数 在 fft 之后 我找到了频率和幅度 我不确定现在需要做什么 例如 我想删除我的最高峰 在绘图上用红点标记 import numpy as np import matplo
  • 如何缩放基于 FFT 的互相关,使其峰值等于 Pearson's rho

    问题描述 FFT 可用于计算两个信号或图像之间的互相关 确定两个信号之间的延迟或滞后A and B 只需定位以下峰值 IFFT FFT A conjugate FFT B 然而 峰值的幅度与各个信号的频谱的幅度相关 从而确定皮尔逊相关系数
  • 我应该从 getFft 看到什么样的输出?

    好吧 我正在努力创建一个 Android 音频可视化应用程序 问题是 我从 getFft 方法中得到的结果与谷歌所说的它应该产生的结果不一致 我一直追溯到 C 源代码 但我对 C 或 FFT 不够熟悉 无法真正理解正在发生的事情 我将尝试包
  • 使用 fftw 和窗函数生成正确的频谱图

    对于一个项目 我需要能够从 WAV 文件生成频谱图 我读过以下应该做的事情 获取N 变换大小 个样本 Apply a window http en wikipedia org wiki Window function功能 使用样本进行快速傅
  • 使用 androids 可视化器类获取可变频率范围

    我想获取智能手机播放的声音的某些频率范围的值 以便我可以通过蓝牙将它们转发到可视化这些范围的设备 这些范围是 0 63Hz63 160赫兹160 400赫兹400 1000赫兹1000 2 500Hz2 500 6 250Hz6 250 1
  • MATLAB 中的反向谱图 A La Aphex Twin

    我正在尝试将图像视为频谱图 从而在 MATLAB 中将图像转换为音频信号就像 Aphex Twin 的歌曲中那样舔窗者 http www bastwood com aphex php 不幸的是 我很难得到结果 这是我现在所拥有的 funct
  • 使用 fft 和 ifft 更改频率而不使用整数

    我知道我可以通过改变变量来改变整数频率shift但我怎样才能改变频率使用带小数位的数字 例如 754 或 1 2345 or 67 456 如果我改变变量 shift 到一个非整数类似的数字5 1 我收到错误下标索引必须是小于 2 31 的
  • 对真实输入数据进行高效的 2D FFT?

    我目前正在使用 opencl 对真实输入数据实现二维 FFT 更具体地说是使用 FFT 的快速 2D 卷积 所以我只需要一些行为足够相似的东西来应用卷积 2D FFT 是在行上使用 1D FFT 然后在列上使用 1D FFT 来实现的 为了
  • 什么是“无为”卷积核

    如果我尝试在频率空间中执行卷积核 什么是 不执行任何操作 的内核 换句话说 如果我在应用内核并在频率空间中对其进行归一化后查看图像 我只想查看原始傅里叶变换 是单位矩阵吗 我的内核是 3x3 Thanks 一个什么都不做的 3x3 内核将是
  • 使用 FFT 进行 Matlab 模板匹配

    我正在努力解决 Matlab 中傅立叶域中的模板匹配问题 这是我的图片 艺术家是 DeviantArt 上的 RamalamaCreatures 我的目标是在负鼠的耳朵周围放置一个边界框 就像这个例子 我使用normxcorr2执行模板匹配
  • 绘制图像的傅里叶变换时出现问题。 “ValueError:x 和 y 不能大于二维,但具有形状 (2592,) 和 (2592, 1, 3)”

    我正在尝试获取图像的 fft 然后使用 matplotlib 绘制该 fft 的 fraq 然而 这个错误信息 ValueError x 和 y 不能大于二维 但具有形状 2592 和 2592 1 3 我尝试像这样重塑我的 np arra
  • Python 中的帕塞瓦尔定理

    我试图掌握 Python 的 fft 功能 我偶然发现的奇怪的事情之一是帕塞瓦尔定理 http en wikipedia org wiki Parseval 27s theorem似乎不适用 因为它现在给出的差异约为 50 而它应该是 0
  • 实时音高检测

    用于实时检测用户歌唱的音调FFT https stackoverflow com questions 1351381 fft problem returns random results and 自相关 https stackoverflo
  • DSP 库 - RFFT - 奇怪的结果

    最近我一直在尝试在我的STM32F4 Discovery评估板上进行FFT计算 然后将其发送到PC 我已经调查了我的问题 我认为我对制造商提供的 FFT 函数做错了 我正在使用 CMSIS DSP 库 现在我一直在用代码生成样本 如果工作正
  • Clojure/Java:用于声音频谱分析的 Java 库? [关闭]

    Closed 此问题正在寻求书籍 工具 软件库等的推荐 不满足堆栈溢出指南 help closed questions 目前不接受答案 我正在寻找一个可以接受大量音频数据并返回给定频带内随时间变化的平均幅度的库 我已经在 comp dsp
  • 使用 FFT 执行音频分析

    我已经被这个问题困扰好几天了 并且浏览了几乎所有相关的 StackOverflow 页面 通过这次活动 我现在对 FFT 是什么及其工作原理有了更深入的了解 尽管如此 我在将其实现到我的应用程序中时遇到了极大的困难 简而言之 我想做的是为我
  • 通过 cuFFT 进行逆 FFT 缩放

    每当我使用 cuFFT 绘制程序获得的值并将结果与 Matlab 的结果进行比较时 我都会得到相同形状的图形 并且最大值和最小值位于相同的点 然而 cuFFT 得到的值比 Matlab 得到的值大得多 Matlab代码是 fs 1000 s

随机推荐

  • 手撕/手写/自己实现 BN层/batch norm/BatchNormalization python torch pytorch

    计算过程 在卷积神经网络中 BN 层输入的特征图维度是 N C H W 输出的特征图维度也是 N C H W N 代表 batch size C 代表 通道数 H 代表 特征图的高 W 代表 特征图的宽 我们需要在通道维度上做 batch
  • 51单片机上连YL69土壤湿度传感器获取的数据在LCD上显示出来

    要做一个项目 被分配到做DS18B20温度传感与YL69土壤湿度传感器在51单片机上用LCD显示屏显示出来 温度传感模块很简单 网上到处都是资料 但是YL69的资料就很少了 特别还是在51单片机上实现 其实懂了原理也还是简单 将传感器的AO
  • 高并发+海量数据下如何实现系统解耦?【上】

    V xin ruyuanhadeng获得600 页原创精品文章汇总PDF 一 写在前面 之前更新过一个 亿级流量系统架构 系列 主要讲述了一个大规模商家数据平台的如下几个方面 如何承载百亿级数据存储 如何设计高容错的分布式架构 如何设计承载
  • 拓端tecdat

    最近我们被客户要求撰写关于偏最小二乘法 PLS 回归的研究报告 包括一些图形和统计输出 本文建立偏最小二乘法 PLS 回归 PLSR 模型 以及预测性能评估 为了建立一个可靠的模型 我们还实现了一些常用的离群点检测和变量选择方法 可以去除潜
  • 全国程序员收入大调查,粒度到省

    2019年五一假期 我没休息 而是统计某招聘网站了全国的程序员工资 总体统计 2019年4月全国招收程序员302303人 2019年4月全国程序员平均工资12807元 工资中位数11500元 其中95 的人的工资介于3750元到32500元
  • (python算法)LeetCode-版本号比较

    第一次笔试 发挥的很糟糕 基础不好是硬伤 碰到了版本号比较这个问题 回来后搜了下 发现在LeetCode里有 正好再仔细研究下 以下是原题 比较两个版本号 version1 和 version2 如果 version1 gt version
  • python怎么一步步调试_PyCharm入门第一步(二)——调试第一个Python应用程序

    第2步 调试您的第一个Python应用程序 找出问题的根源 PyCharm报告运行时错误 a ZeroDivisionError 深入研究一下代码 找出问题所在 这里可以使用PyCharm调试器来查看代码中发生了什么 要开始调试 您必须先设
  • 【珍藏版】 2012Java开发工程师必备精品资料(115个)

    Java应用广泛 涉及个人PC 数据中心 游戏控制台 科学超级计算机 移动电话和互联网等领域 同时拥有全球最大的开发者专业社群 小弟精心整理了115个精品资料 包括11个Java开发专题和104个热门资源 网上的资料众多 参差不齐 然而这批
  • PHPBONE使用问题集--.Net直接POST数据被过滤

    当 NET用POST发送数据到服务端时 发现 加号全被过滤成空格了 以为是PHPBONE的问题 查了半天代码也没发现哪有异常 但是以前也遇到过 也的确是处理过 只是不记得是怎么处理的了 无耐翻出以前的程序查找了一番 结果发现是编码问题 把数
  • 2021-07-18

    JQuery之DOM操作 1 创建节点及结点属性 1 DOM创建节点及结点属性 创建流程比较简单 大体如下 创建节点 常见的 元素 属性和文本 添加节点的一些属性 加入到文档中 流程中涉及的一点方法 创建元素 document create
  • 哲学家问题(死锁问题)

    1 问题描述 有五个哲学家绕着圆桌坐 每个哲学家面前有一盘面 两人之间有一支筷子 这样每个哲学家左右各有一支筷子 哲学家有2个状态 思考或者拿起筷子吃饭 如果哲学家拿到一只筷子 不能吃饭 直到拿到2只才能吃饭 并且一次只能拿起身边的一支筷子
  • git从某个分支创建新分支

    如题 记录一下从某个分支创建新分支的方法 如从dev分支创建一个test分支 第一步 切换到你指定的分支 如我要从dev上拉一个分支 代码一模一样 git checkout dev 第二步 拉取dev的最新代码 git pull 第三步 在
  • Android Bitmap加载内存占用彻底分析

    背景 在某个版本应用上线后 偶然测得首页占用的内存非常的大而且一直不能回收掉 经过一轮的排查后最终确定是3张图片引起的 当时每张图片占用了将近20m内存 当时紧急处理好后还一直惦记着此事 后来对Android加载Bitmap的内存占用作了彻
  • Android系统源代码的下载与编译

    http www jianshu com p aeaceda41798
  • UVa 12955 Factorial

    Problem uva onlinejudge org index php option com onlinejudge Itemid 8 page show problem problem 4834 开始想多了 想着不能简单贪心 要用dp
  • C# Task异步编程

    Task任务用法 Task用的是线程池 线程池的线程数量的有上限的 这个可以通过ThreadPool修改 我们经常会用到task run new task 和task factory startnew方法来创建任务 Task Factory
  • 检查HDFS块状态

    hadoop集群运行过程中 节点的块状态或者上下线节点时集群都会受影响 如何查看当前的hdfs的块的状态 hadoop1 x时候的命令 hadoop2 x也可使用 hadoop fsck 在hadoop2 0之后 可以使用新命令 hdfs
  • 关于 JavaScript 中的 Promises

    在 JavaScript 中 Promise 是一个对象 它表示一个可能还不可用 但会在未来解决的值 Promises 用于处理异步操作 例如发出网络请求或访问数据库 其中结果不是立即可用的 如果你准备好了 我想开始我们的冒险 承诺如何运作
  • vc2010中开始执行不调试灰的_Intellij IDEA调试功能使用总结

    这段时间一直在使用Intellij IDEA 今天把调试区工具的使用方法记录于此 先编译好要调试的程序 1 设置断点 选定要设置断点的代码行 在行号的区域后面单击鼠标左键即可 2 开启调试会话 点击红色箭头指向的小虫子 开始进入调试 IDE
  • 快速傅里叶变换(FFT)

    前言 在学习FFT过程中看了很多博客 但发现在看博客的时候博客上的内容大多都晦涩难懂 于是乎想自己写一篇博客来记录一下自己学习的心得体会 知其源 先来讲讲FFT的起源 快速傅里叶变换是1965年由J W 库利和T W 图基提出的 采用这种算