Matlab实现FFT变换

news2025/1/29 14:00:53

Matlab实现FFT变换

文章目录

    • Matlab实现FFT变换
      • 原理
      • 实现
        • 手算验证
        • 简单fft变换和频谱
        • 求取功率谱
      • 结论

在信号处理中,快速傅里叶变换(FFT)是一种非常常见的频域分析方法。本文将介绍如何使用Matlab实现FFT变换,并通过Matlab代码演示实际输出结果。

原理

FFT是一种计算离散傅里叶变换(DFT)的快速算法。DFT将时域上的信号转换为频域上的信号,可以用以下公式表示:

X k = ∑ n = 0 N − 1 x n e − i 2 π k n / N X_k=\sum_{n=0}^{N-1}x_n e^{-i2\pi kn/N} Xk=n=0N1xnei2πkn/N

其中, x n x_n xn是时域上的信号序列, X k X_k Xk是频域上的信号序列, k k k为频率编号( 0 ≤ k < N 0\leq k < N 0k<N), N N N为信号长度。

FFT算法通过分治策略将DFT算法的计算复杂度从 O ( N 2 ) O(N^2) O(N2)降低到 O ( N l o g 2 N ) O(Nlog_2N) O(Nlog2N),从而实现了在计算机上快速计算DFT的目的。

实现

手算验证

在这里,我们将给出一个简单的例子来说明如何使用Matlab进行FFT变换。我们首先生成一个简单的数组:

x = [1, 2, 3, 4, 5, 6, 7, 8];

接下来,我们使用Matlab内置的fft函数对这个数组进行FFT变换:

X = fft(x);

这个操作会返回一个和输入数组长度相同的复数数组。我们可以使用Matlab的disp函数打印出这个数组:

disp(X);

这个例子的输出结果如下:

   36.0000 + 0.0000i
   -4.0000 + 9.6569i
   -4.0000 + 4.0000i
   -4.0000 + 1.6569i
   -4.0000 + 0.0000i
   -4.0000 - 1.6569i
   -4.0000 - 4.0000i
   -4.0000 - 9.6569i

可以看到,输出结果是一个长度为8的复数数组。

为了验证FFT的正确性,我们可以手动计算这个输入数组的FFT结果,然后将结果与Matlab计算的结果进行比较。FFT算法的计算过程可以用以下公式表示:

X k = ∑ n = 0 N − 1 x n e − i 2 π k n / N X_k=\sum_{n=0}^{N-1}x_n e^{-i2\pi kn/N} Xk=n=0N1xnei2πkn/N

其中, x n x_n xn是时域上的信号序列, X k X_k Xk是频域上的信号序列, k k k为频率编号( 0 ≤ k < N 0\leq k < N 0k<N), N N N为信号长度。

对于输入数组 x = [ 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 ] x=[1,2,3,4,5,6,7,8] x=[1,2,3,4,5,6,7,8],我们有 N = 8 N=8 N=8。因此, X 0 X_0 X0的计算公式为:

X 0 = ∑ n = 0 7 x n e − i 2 π 0 n / 8 = 36 X_0 = \sum_{n=0}^{7}x_n e^{-i2\pi 0n/8}=36 X0=n=07xnei2π0n/8=36

接下来,我们可以计算 X 1 X_1 X1

X 1 = ∑ n = 0 7 x n e − i 2 π 1 n / 8 = − 4 + 9.6569 i X_1 = \sum_{n=0}^{7}x_n e^{-i2\pi 1n/8}=-4+9.6569i X1=n=07xnei2π1n/8=4+9.6569i

以此类推,我们可以计算出所有的 X k X_k Xk。最终结果应该与Matlab计算的结果一致。

简单fft变换和频谱

以下是一个简单的Matlab代码实现FFT变换:

% 生成测试信号
Fs = 1000;      % 采样频率
t = 0:1/Fs:1-1/Fs;   % 时间向量
x = 1*sin(2*pi*100*t); % 信号

% 绘制信号图
subplot(2,1,1);
plot(t,x);
title('信号');
xlabel('时间 (s)');
ylabel('幅度');

% 计算FFT
N = length(x);
X = fft(x);
f = Fs*(0:(N/2))/N;

% 绘制FFT图
subplot(2,1,2);
plot(f,abs(X(1:N/2+1)));
title('FFT');
xlabel('频率 (Hz)');
ylabel('幅度');

image-20230308202228276

在这个例子中,我们生成了一个频率为100Hz的正弦信号。我们使用Matlab的fft函数计算FFT,并将结果绘制成幅度谱。注意,在绘制幅度谱时,我们只绘制了频率为正的一半,因为FFT算法输出的结果是对称的。

求取功率谱

通过FFT变换可以得到信号的幅度谱,但是为了更好地了解信号特性,我们通常需要求取信号的功率谱密度。功率谱密度描述了信号在不同频率下的功率分布情况。

求取功率谱的方法是,将信号进行FFT变换后,将每个频率上的幅度平方除以信号长度,并乘以一个系数,即可得到功率谱密度。具体公式如下:

P k = 2 ∣ X k ∣ 2 N P_k=\frac{2|X_k|^2}{N} Pk=N2∣Xk2

其中, P k P_k Pk是频率为 k k k的功率谱密度, X k X_k Xk是频率为 k k k的信号幅度, N N N是信号长度。

以下是一个简单的Matlab代码实现求取功率谱:

% 生成测试信号
Fs = 1000;      % 采样频率
t = 0:1/Fs:1-1/Fs;   % 时间向量
x = 1*sin(2*pi*100*t); % 信号

% 绘制信号图
subplot(2,1,1);
plot(t,x);
title('信号');
xlabel('时间 (s)');
ylabel('幅度');

% 计算FFT
N = length(x);
X = fft(x);
f = Fs*(0:(N/2))/N;

% 计算功率谱
P = (2*abs(X(1:N/2+1)).^2)/N;

% 绘制功率谱图
subplot(2,1,2);
plot(f,P);
title('功率谱密度');
xlabel('频率 (Hz)');
ylabel('功率');

image-20230308201245085

注意,我们在计算功率谱时,使用了一个系数 2 2 2,这是因为我们只绘制了频率为正的一半,而实际上信号的功率谱是对称的。

结论

本文介绍了如何使用Matlab实现FFT变换,并求取信号的功率谱密度。通过FFT变换,我们可以将信号从时域转换到频域,进一步了解信号的特性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/396651.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SAP ABAP 深度解析Smartform打印特殊符号等功能

ABAP 开发人员可以在 Smartform 输出上显示 SAP 图标或 SAP 符号。例如,需要在 SAP Smart Forms 文档上显示复选框形状的输出。SAP Smartform 文档上可以轻松显示空复选框、标记复选框以及 SAP 图标等特殊符号。 在 SAP Smartform 文档中添加一个新的文本节点。 1. 单击“更…

开发一款系统软件的流程步骤是什么

在如今的数字化时代&#xff0c;软件开发成为了一个重要的行业。无论是大型企业还是小型创业公司&#xff0c;软件开发都是不可或缺的一环。在本文中&#xff0c;我将介绍一些网上常见的软件开发步骤&#xff0c;以便开发者能够更好地理解和实践。1、需求分析需求分析是开发系统…

基于transformer的多帧自监督深度估计 Multi-Frame Self-Supervised Depth with Transformers

Multi-Frame Self-Supervised Depth with Transformers基于transformer的多帧自监督深度估计0 Abstract 多帧深度估计除了学习基于外观的特征外&#xff0c;也通过特征匹配利用图像之间的几何关系来改善单帧估计。我们采用深度离散的核极抽样来选择匹配像素&#xff0c;并通过一…

基于Jeecgboot前后端分离的ERP系统开发代码生成(六)

商品信息原先生成的不符合要求&#xff0c;重新生成&#xff0c;包括一个附表商品价格信息表 一、采用TAB主题一对多的模式 因为主键&#xff0c;在online表单配置是灰的&#xff0c;所以不能进行外键管理&#xff0c;只能通过下面数据库进行关联录入&#xff0c;否则online界面…

案例19-遇见问题的临时解决方案和最终解决方案

目录1、背景介绍2、两种解决方案的概念1、临时解决方案&#xff1a;2、最终解决方案&#xff1a;3、排查问题过程4、总结站在用户的角度思考作为软件开发者5、升华1、背景介绍 首先说明这是系统很早之前的时候的一个功能&#xff0c;当时和学习通还有很强的耦合关系。在学习通…

研究链表空间销毁问题

&#x1f4af;&#x1f4af;&#x1f4af; 1.研究链表空间销毁问题 当链表使用完后&#xff0c;需要将链表销毁&#xff0c;那么该如何销毁呢&#xff1f; void SLTDestroy(SLTNode* phead)//销毁单链表 {SLTNode* cur phead;while(cur){free(cur);cur cur->next;} }你…

Linux下Nginx安装使用

一、下载解压nginx # 进入要放安装包的目录 cd /opt/software # 下载安装包 wget https://nginx.org/download/nginx-1.20.2.tar.gz # 解压缩 tar -zxvf nginx-1.20.2.tar.gz -C /opt/modules # 进入解压后的目录 cd /opt/modules/nginx-1.20.2/二、安装nginx 1、安装编译器 …

剑指 Offer II 021. 删除链表的倒数第 n 个结点

题目链接 剑指 Offer II 021. 删除链表的倒数第 n 个结点 mid 题目描述 给定一个链表&#xff0c;删除链表的倒数第 n个结点&#xff0c;并且返回链表的头结点。 示例 1&#xff1a; 输入&#xff1a;head [1,2,3,4,5], n 2 输出&#xff1a;[1,2,3,5] 示例 2&#xff1a; …

MySQL的多表操作

多表关系 介绍 实际开发中&#xff0c;一个项目通常需要很多张表才能完成。例如&#xff1a;一个商城项目就需要分类表(category)、商品表(products)、 订单表(orders)等多张表。且这些表的数据之间存在一定的关系&#xff0c;接下来我们将在单表的基础上&#xff0c;一起学习…

DolphinDB 机器学习在物联网行业的应用:实时数据异常率预警

数据异常率预警在工业安全生产中是一项重要工作&#xff0c;对于监控生产过程的稳定性&#xff0c;保障生产数据的有效性&#xff0c;维护生产设备的可靠性具有重要意义。随着大数据技术在生产领域的深入应用&#xff0c;基于机器学习的智能预警已经成为各大生产企业进行生产数…

logback无法删除太久远的日志文件?logback删除日志文件源码分析

logback无法删除太久远的日志文件&#xff1f;logback删除日志文件源码分析 最近发现logback配置滚动日志&#xff0c;但是本地日志文件甚至还有2年前的日志文件&#xff0c;服务器是却是正常的&#xff01; 网上搜索了一波没有发现&#xff0c;只找到说不能删除太久远的旧日志…

Leetcode. 21 合并两个有序列表

尾插 核心思路&#xff1a;依次比较 &#xff0c;取经过比较后较小值进行尾插 cur1 指向list1 ,cur 2指向list2 ,当cur1走完list1 或者cur2 走完list2 后停止 如果cur1走完list1 ,可以将cur2 整个拿下来尾插 如果cur2走完list2 ,可以将cur1 整个拿下来尾插 特殊情况 &#xff1…

c# 32位程序突破2G内存限制

起因 在开发过程中&#xff0c;由于某些COM组件只能在32位程序下运行&#xff0c;程序不得不在X86平台下生成。而X86的32位程序默认内存大小被限制在2G。由于程序中可能存在大数量处理&#xff0c;期间对象若没有及时释放或则回收&#xff0c;内存占用达到了1.2G左右&#xff…

瀑布开发与敏捷开发的区别,以及从瀑布转型敏捷项目管理的5大注意事项

事实证明&#xff0c;瀑布开发管理模式并不适合所有的软件项目&#xff0c;但敏捷项目管理却对大多数项目有效。那么当团队选择转型敏捷的时候有哪些因素必须注意&#xff1f;敏捷开发最早使用者大多是小型、独立的团队&#xff0c;他们通常致力于小型、独立的项目。正是他们的…

Keepalive+LVS群集部署

KeepaliveLVS群集部署一、Keepalive概述1、什么是Keepalive2、Keepalive工作原理3、Keepalive主要模块及作用4、Keepalived 服务重要功能&#xff08;1&#xff09;管理 LVS 负载均衡软件&#xff08;2&#xff09;支持故障自动切换&#xff08;3&#xff09;实现 LVS 负载调度…

SpringBoot下的Spring框架学习(tedu)——day03——Spring DI、SpringAOP JDK动态代理

SpringBoot下的Spring框架学习&#xff08;tedu&#xff09;——day03——Spring DI、SpringAOP JDK动态代理 目录SpringBoot下的Spring框架学习&#xff08;tedu&#xff09;——day03——Spring DI、SpringAOP JDK动态代理1. Spring的依赖注入1.1 依赖注入案例1.1.1 定义接口…

Python的运算符

Python支持多种运算符&#xff0c;下表大致按照优先级从高到低的顺序列出了所有的运算符&#xff0c;运算符的优先级指的是多个运算符同时出现时&#xff0c;先做什么运算然后再做什么运算。除了我们之前已经用过的赋值运算符和算术运算符&#xff0c;我们稍后会陆续讲到其他运…

解决launch:program .exe does not exist

二. 程序的运行和调试 1.launch.json 复制下列代码至launch.json&#xff0c;并根据指导做出相对/绝对路径修改 用 IntelliSense 了解相关属性。 {// 使用 IntelliSense 了解相关属性。 // 悬停以查看现有属性的描述。// 欲了解更多信息&#xff0c;请访问: https://go.micros…

虹科教您 | 在Linux环境下安装PCAN View及通讯测试指南

应用简介 PCAN-View软件是一款简化的CAN监视软件&#xff0c;可用于显示、发送、和记录CAN数据通讯。报文可手动和定期发送&#xff0c;用户可设置比特率。在处理期间显示总线系统错误和CAN硬件的存储器过满。示踪功能可用于记录和保存CAN数据通讯。本文档的作用在于&#xff…

Python进阶-----面向对象8.0(反射---对象属性方法操作)

目录 前言&#xff1a; 反射 1.getattr() 函数 2.setattr() 函数 3.hasattr() 函数 4.delattr() 函数 前言&#xff1a; 众所周知&#xff0c;每一个实例化对象都是有实例化属性或者方法的&#xff0c;之前我们对实例化对象进行属性方法查询、删除或者添加都是直接去通过…