0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

看完学会速傅立叶变换FFT

jf_78858299 来源:ACM算法日常 作者: Xiejiadong 2023-05-05 09:48 次阅读

FFT 即快速傅立叶变换。在很多计算机领域都用用处,例如数字图像处理、计算机网络。但他在算法竞赛中主要是用于多项式和生成函数相关的题目。

多项式

表达方式

简介

  • 系数表达式,即。
  • 坐标形式。每一个坐标用表示。显然,为了能够表示一个确定的多项式,需要个不同的坐标来表示。

比较

  • 对于系数表示,多项式加法的时间复杂度是,多项式乘法的时间复杂度是。
  • 对于点值表示,多项式加法的时间复杂度同样是,但是乘法的时间复杂度就是(因为多项式乘法以后最高项次数为,我们只需要个坐标表示)。

思考

这样一来,我们就有一个想法,多项式乘法,是不是可以利用坐标表示做多项式乘法特别快这点来优化算法。

于是需要解决的最大的问题就是,多项式两种表示方法之间的互相转换。

求值朴素做法的时间复杂度是,即随便选取一个值带入,暴力计算。

差值朴素的做法时间复杂度是,即根据坐标进行高斯消元。

在介绍 FFT 之前,得先学习 DFT (离散傅里叶变换)算法。

DFT

由于对一个多项式的点值表达式的取是任意的,所以好的取法可能会使一个算法产生本质性的蜕变。

我们选取次单位复根作为来取值。

单位复根

,这个方程的复数根为次单位根。

单位的个单位根分别为。

个单位根在复平面的坐标表示为,我们将这个记为。在复平面上形象的表示的话,就是下图:

图片

单位根在多项式的应用

我们将个单位根带入多项式可以得到个因变量结果,记为。

我们将个单位根的倒数作为自变量带入由作为系数的多项式,可以得到。

而当的时候,它为,其余时候,它为(通过等比数列求和)。于是有。

于是可以发现,将个单位根的倒数带入变换后的多项式,可以得到原来的多项式。

这样一来,我们利用个单位根的性质,完成了多项式两种表示方式之间的转换。

DFT算法

有了的取值,我们就可以得到的取值了。

直接暴力计算,两个方向转换的时间复杂均为。

FFT

那么 FFT 算法是如何优化计算这一过程的?利用分治。

我们把一个多项式的计算分为偶数项的计算和奇数项的计算:

也就是说, FFT 的思想就是不断得把一个多项式拆分成奇数多项式和偶数多项式,然后合并两个多项式的信息

也就是说,如果我们已经得到了和,我们只需要就可以得到了。

每次都能把多项式的长度减小一半,于是时间复杂度就是。

模版

C++ 是自带了复数 stl 的,即:

#include

但是常数大,跑得慢,不如自己重载的好。

  • 下面的模版必须要保证是的整数次幂。
typedef double LD;
const LD PI = acos(-1);
struct C {
    LD r, i;
    C(LD r = 0, LD i = 0): r(r), i(i) {}
};
C operator + (const C& a, const C& b) {
    return C(a.r + b.r, a.i + b.i);
}
C operator - (const C& a, const C& b) {
    return C(a.r - b.r, a.i - b.i);
}
C operator * (const C& a, const C& b) {
    return C(a.r * b.r - a.i * b.i, a.r * b.i + a.i * b.r);
}

void FFT(C x[], int n, int p) {
    for (int i = 0, t = 0; i < n; ++i) {
        if (i > t) swap(x[i], x[t]);
        for (int j = n >> 1; (t ^= j) < j; j >>= 1);
    }
    for (int h = 2; h <= n; h <<= 1) {
        C wn(cos(p * 2 * PI / h), sin(p * 2 * PI / h));
        for (int i = 0; i < n; i += h) {
            C w(1, 0), u;
            for (int j = i, k = h >> 1; j < i + k; ++j) {
                u = x[j + k] * w;
                x[j + k] = x[j] - u;
                x[j] = x[j] + u;
                w = w * wn;
            }
        }
    }
    if (p == -1)
        FOR (i, 0, n)
            x[i].r /= n;
}

void conv(C a[], C b[], int n) {
    FFT(a, n, 1);
    FFT(b, n, 1);
    FOR (i, 0, n)
        a[i] = a[i] * b[i];
    FFT(a, n, -1);
}

例题

A * B II

https://acm.ecnu.edu.cn/problem/3007/

大整数相乘。

把每一位都看成是多项式其中一项的系数,那么大数最后的结果也就是多项式乘法系数的结果。

要进位处理。

Hnoi2017 礼物

显然是要计算的最小值,其中$0≤x

展开这个式子,

除了,其他的和与相关的项都可以在的时间内算出了

那么配个方,就可以求出最小值了,而是固定的

现在的问题就是求,我们可以用FFT来解决

如果我们把多项式倒置,我们就能发现式子的和的下标和可以相同,我们可以利用多项式乘法同时算出卷积。

设,那么,这样就可以用FFT算出来了

总的时间复杂度

#include
#define inf 0x3fffffff
using namespace std;

typedef double LD;
const LD PI=acos(-1);
struct C
{
    LD r,i;
    C(LD r=0,LD i=0):r(r),i(i){}
};
C operator + (const C& a, const C& b){
    return C(a.r+b.r,a.i+b.i);
}
C operator - (const C& a, const C& b){
    return C(a.r-b.r,a.i-b.i);
}
C operator * (const C& a, const C& b){
    return C(a.r*b.r-a.i*b.i,a.r*b.i+a.i*b.r);
}
 
void FFT(C x[],int n,int p)
{
    for (int i=0,t=0;i
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 图像处理
    +关注

    关注

    27

    文章

    1292

    浏览量

    56762
  • FFT
    FFT
    +关注

    关注

    15

    文章

    434

    浏览量

    59396
  • 计算机网络
    +关注

    关注

    3

    文章

    337

    浏览量

    22173
  • 傅立叶变换
    +关注

    关注

    3

    文章

    105

    浏览量

    32401
收藏 人收藏

    评论

    相关推荐

    快速傅立叶变换(FFT)算法实验

    本帖最后由 mr.pengyongche 于 2013-4-30 02:23 编辑 快速傅立叶变换(FFT)算法实验一、摘
    发表于 12-21 10:54

    如何使用快速傅立叶变换FFT)的8590 C/E/L系列频谱分析仪中的FFT函数?

    本产品说明说明了如何使用快速傅立叶变换FFT)的8590 C/E/L系列频谱分析仪中的FFT函数。FFT通过提供智能用户界面简化了AM分析
    发表于 04-04 16:50

    浅懂示波器FFT快速傅立叶变换功能及运用

    大多数示波器上都有个FFT功能,也叫快速傅立叶变换,但很多人不了解这个功能是做什么用的,百度以后又会遇到各种各样的高数公式,看的一头雾水,遂而放弃这块知识。我们来看百度百科的解释:FFT
    发表于 01-14 17:00

    示波器FFT快速傅立叶变换不会用?看完这篇帖子,我彻底悟了

    大多数示波器上都有个FFT功能,也叫快速傅立叶变换,但很多人不了解这个功能是做什么用的,百度以后又会遇到满屏的高数公式,看得一头雾水,继而以放弃告终。先来看看百度百科对FFT的解释:
    发表于 09-22 13:42

    快速傅立叶变换开发指南

    快速傅立叶变换开发指南:The Xilinx® LogiCORE™ IP Fast Fourier Transform (FFT) is a computationally
    发表于 12-31 15:19 35次下载

    快速傅立叶变换FFT)的Nios II实现

    快速傅立叶变换FFT)的Nios II实现 随着数字电子技术的发展,数字信号处理的理论和技术广泛地应用于通讯、语音处理、计算机和多媒体等领域。快速傅里叶
    发表于 02-09 09:38 81次下载

    基于FPGA的快速傅立叶变换

    摘要:在对FFT(快速傅立叶变换)算法进行研究的基础上,描述了用FPGA实现FFT的方法,并对其中的整体结构、蝶形单元及性能等进行了分析。
    发表于 06-20 14:13 1104次阅读

    1024点FFT快速傅立叶变换

    Xilinx FPGA工程例子源码:1024点FFT快速傅立叶变换
    发表于 06-07 14:13 33次下载

    Xilinx 的IP:1024点FFT快速傅立叶变换

    Xilinx FPGA工程例子源码:Xilinx 的IP:1024点FFT快速傅立叶变换
    发表于 06-07 15:07 51次下载

    DSP进行浮点快速傅立叶变换剖析

    前言本文目的是演示如何使用STM32F30x 内部的DSP 进行浮点快速傅立叶变换FFT),为联系实际应用
    的头像 发表于 09-18 06:44 9525次阅读

    简述FPGA的快速傅立叶变换

    摘要:在对FFT(快速傅立叶变换)算法进行研究的基础上,描述了用FPGA实现FFT的方法,并对其中的整体结构、蝶形单元及性能等进行了分析。 傅立叶
    的头像 发表于 05-27 11:21 2245次阅读
    简述FPGA的快速<b class='flag-5'>傅立叶</b><b class='flag-5'>变换</b>

    傅立叶变换是怎么变换傅立叶的理解

    关于傅立叶变换,无论是书本还是在网上可以很容易找到关于傅立叶变换的描述,但是大都让人很难理解太过抽象,尽是一些让人看了就望而生畏的公式的罗列。 要理解
    的头像 发表于 08-25 11:25 4907次阅读

    我印象中的快速傅里叶变换 (FFT)

    首先,FFT是离散傅立叶变换 (DFT) 的快速算法,那么说到FFT,我们自然要先讲清楚傅立叶变换
    的头像 发表于 05-05 09:57 1180次阅读
    我印象中的快速傅里叶<b class='flag-5'>变换</b> (<b class='flag-5'>FFT</b>)

    浅懂示波器FFT快速傅立叶变换功能及运用

    大多数示波器上都有个FFT功能,也叫快速傅立叶变换,但很多人不了解这个功能是做什么用的,百度以后又会遇到各种各样的高数公式,看的一头雾水,遂而放弃这块知识。我们来看百度百科的解释:FFT
    的头像 发表于 11-08 15:01 6877次阅读
    浅懂示波器<b class='flag-5'>FFT</b>快速<b class='flag-5'>傅立叶</b><b class='flag-5'>变换</b>功能及运用

    如何使用SBench 6对数字化仪采集信号进行处理?(三)——快速傅立叶变换FFT

    上一篇文章介绍了德思特SBench 6的平均运算功能。本章将继续为大家介绍SBench 6的快速傅立叶变换FFT)。
    的头像 发表于 01-23 10:38 601次阅读
    如何使用SBench 6对数字化仪采集信号进行处理?(三)——快速<b class='flag-5'>傅立叶</b><b class='flag-5'>变换</b>(<b class='flag-5'>FFT</b>)