数字信号处理基础(二):FFT和IFFT的使用以及详细分析代码书写思路

1. fft和ifft的原理

1.1 fft

fft是快速傅里叶变换,是MATLAB中计算信号频谱的函数,使用方法是fft(x),直接对信号x进行fft计算。
由于fft函数计算信号的频谱是0-fs范围的,但是我们更习惯观察的信号的双边频谱,即-fs/2-fs/2范围的频谱,因此常用fftshift(x)函数将0-fs范围的信号频谱搬移到-fs/2-fs/2范围,也就是将原来0-fs范围中的fs/2-fs范围的频谱搬移到-fs/2-0范围处。

1.2 ifft

ifft是快速逆傅里叶变换,是MATLAB中求信号的逆傅里叶变换的函数,使用方法是ifft(x)。

2. 书写代码思路

1.养成良好习惯,在代码的最上面书写三件套代码,该清除的清除。

clear all;                  % 清除所有变量
close all;                  % 关闭所有窗口
clc;                        % 清屏

2.确定各个信号的频率以及信号的最高频率,本例中信号是正弦信号和余弦信号之和,分别确定各个信号频率,以及信号的最高频率,这里是3000Hz

fc1 = 1000;                 % 信号1的中心频率
fc2 = 3000;                 % 信号2的中心频率

3.根据信号的最高频率和奈奎斯特采样定理确定采样频率(大于信号最高频率的2倍即可),本人为了时域信号的平滑,通常将采样频率设置为最高频率的30倍左右。

fs = 100000;                % 采样频率,满足奈奎斯特采样定理即可,这里为信号频率的33倍原因是为了采样点数多,时域波形平滑

4.根据采样频率求得采样间隔,即采样频率的倒数

dt = 1/fs;                  % 采样间隔

5.确定仿真信号时长,这个一般根据实际情况来定,例如在通信系统中,首先需要确定码元周期,然后确定码元个数,两者相乘即是仿真时长,在本例中设置的时长是0.1。

T = 0.1;                    % 采样总时长

6.确定采样点数,为采样时长除以采样间隔。

N = T*fs;                   % 采样点数,N=T/dt=T*fs

7.确定仿真时间变量,从0开始,以采样间隔为间隔,形成1*N的行向量

t = 0:dt:(N-1)*dt;          % 时间行向量,间隔为dt

8.产生信号,并且绘制图像。

sig = sin(2*pi*fc1*t)+cos(2*pi*fc2*t);% 信号
figure(1);                  % 绘制第1幅图
plot(t,sig,'LineWidth',2);  % 绘制信号波形
axis([0 0.002 -2 2]);       % 坐标轴范围设置
xlabel('时间/t');           % 横坐标标签
ylabel('幅值');             % 纵坐标
title('原始信号');          % 标题
grid on;                    % 打开网格线

9.对信号进行fft,fft三步走

sig_f = fft(sig);           % 对信号进行fft
sig_f_2 = abs(sig_f);       % fft后取绝对值得到幅度响应
sig_f_2 = fftshift(sig_f_2);%0-fs频谱移动到-fs/2-fs/2

10.确定频谱分辨率fs/N=1/T

df = 1/T;                   % 频谱分辨率fs/N=1/T

11.确定频谱范围,即-fs/2-fs/2,间隔为频谱分辨率

f=(-N/2:N/2-1)*df;          % 频率范围

12.绘制频谱

figure(2);                  % 绘制第2幅图
plot(f,sig_f_2,'LineWidth',2);% 绘制信号的频谱
axis([-5000 5000 -inf inf]);% 坐标轴范围设置
xlabel('频率/Hz');          % 横坐标标签
ylabel('幅值');             % 纵坐标
title('原始信号的频谱');    % 标题
grid on;                    % 打开网格线

13.对信号做ifft,这里是对fft后的信号sig_f进行ifft,而不是对进行fft三步走后的信号sig_f_2进行ifftshift,然后进行ifft,因为三步走中有一步是取绝对值,通过ifftshift然后ifft无法准确还原出原始信号,因此为了还原出原始信号,应该对fft后的信号sig_f进行ifft,这也是为什么要三步走的原因,而不采用一行生成的代码fftshift(abs(fft(x)))。

sig_ifft = ifft(sig_f);     % 对fft后的信号进行ifft还原成时域信号

14.绘制还原后信号图像

plot(t,sig_ifft,'LineWidth',2);% 绘制还原后的时域信号
axis([0 0.002 -2 2]);       % 坐标轴范围设置
xlabel('时间/t');           % 横坐标标签
ylabel('幅值');             % 纵坐标
title('还原后信号');        % 标题
grid on;                    % 打开网格线

3. 完整代码

clear all;                  % 清除所有变量
close all;                  % 关闭所有窗口
clc;                        % 清屏
%% 基本参数
fc1 = 1000;                 % 信号1的中心频率
fc2 = 3000;                 % 信号2的中心频率
fs = 100000;                % 采样频率,满足奈奎斯特采样定理即可,这里为信号频率的33倍原因是为了采样点数多,时域波形平滑
dt = 1/fs;                  % 采样间隔
T = 0.1;                    % 采样总时长
N = T*fs;                   % 采样点数,N=T/dt=T*fs
t = 0:dt:(N-1)*dt;          % 时间行向量,间隔为dt
sig = sin(2*pi*fc1*t)+cos(2*pi*fc2*t);% 信号
figure(1);                  % 绘制第1幅图
plot(t,sig,'LineWidth',2);  % 绘制信号波形
axis([0 0.002 -2 2]);       % 坐标轴范围设置
xlabel('时间/t');           % 横坐标标签
ylabel('幅值');             % 纵坐标
title('原始信号');          % 标题
grid on;                    % 打开网格线
sig_f = fft(sig);           % 对信号进行fft
sig_f_2 = abs(sig_f);       % fft后取绝对值得到幅度响应
sig_f_2 = fftshift(sig_f_2);%0-fs频谱移动到-fs/2-fs/2
df = 1/T;                   % 频谱分辨率fs/N=1/T
f=(-N/2:N/2-1)*df;          % 频率范围
figure(2);                  % 绘制第2幅图
plot(f,sig_f_2,'LineWidth',2);% 绘制信号的频谱
axis([-5000 5000 -inf inf]);% 坐标轴范围设置
xlabel('频率/Hz');          % 横坐标标签
ylabel('幅值');             % 纵坐标
title('原始信号的频谱');    % 标题
grid on;                    % 打开网格线
sig_ifft = ifft(sig_f);     % 对fft后的信号进行ifft还原成时域信号
figure(3);                  % 绘制第3幅图
plot(t,sig_ifft,'LineWidth',2);% 绘制还原后的时域信号
axis([0 0.002 -2 2]);       % 坐标轴范围设置
xlabel('时间/t');           % 横坐标标签
ylabel('幅值');             % 纵坐标
title('还原后信号');        % 标题
grid on;                    % 打开网格线

4. 结果图

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

  • 20
    点赞
  • 110
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
### 回答1: c实现任意点数的FFT及IFFT是指在数字信号处理中,使用C语言编程实现对任意点数的信号进行快速傅里叶变换(FFT)和逆傅里叶变换(IFFT)。 FFT是一种将时域信号转换为频域信号的算法,可以将信号分解成不同频率的成分。而IFFT则是将频域信号重新恢复为时域信号。 实现任意点数的FFT及IFFT的关键是对输入信号进行零填充。零填充是将输入信号的长度扩展为2的幂次方,以便进行FFT计算。具体步骤如下: 1. 首先确定输入信号的长度,记为N。 2. 找到大于N的最小的2的幂次方,记为M。例如,如果N=9,则M=16。 3. 创建一个长度为M的数组,将输入信号拷贝到该数组中,并将其余部分填补为0。 4. 对这个数组进行FFT计算,得到频域信号。 5. 如果需要进行IFFT计算,则对频域信号进行IFFT计算,并得到恢复的时域信号。 在C语言中,可以使用库函数或自己实现FFT和IFFT算法。常用的库函数有FFTW、KissFFT等,它们都提供了对任意点数的FFT和IFFT计算。 总之,实现任意点数的FFT及IFFT需要对输入信号进行零填充,并采用合适的算法进行计算。通过使用C语言编程,可以轻松地实现这些功能。 ### 回答2: c 实现任意点数的fft及ifft(数字信号处理 fft).zip是一个压缩文件,其中包含有关数字信号处理中快速傅里叶变换(FFT)以及逆傅里叶变换(IFFT)的代码实现。 FFT是一种高效的算法,可以将一个信号从时间域转换到频率域。它广泛应用于信号处理、通信和图像处理等领域。在FFT算法中,信号被分解为一系列的频率成分,这些频率成分可以通过完成一系列乘法和加法操作来计算。FFT算法可以显著减少运算量,提高计算速度。 IFFTFFT的逆变换,可以将信号从频率域转换回时间域。它与FFT具有相似的算法思想,但是计算步骤相反。通过IFFT,我们可以恢复原始信号,从而实现信号的逆变换。 c 实现任意点数的fft及ifft(数字信号处理 fft).zip文件中的代码实现了FFT和IFFT算法,可以处理任意点数的信号。您可以使用这些代码来实现您自己的信号处理任务,例如频谱分析、滤波和信号重构等。 要使用这个压缩文件,您可以下载并解压缩它。然后,您可以查看代码示例,并根据自己的需求进行修改和使用。建议您阅读附带的文档,了解如何正确使用这些算法,并理解算法的原理。 通过使用c 实现任意点数的fft及ifft(数字信号处理 fft).zip中提供的代码和算法,您可以更好地理解和应用傅里叶变换在数字信号处理中的重要性和作用。 ### 回答3: c 实现任意点数的fft及ifft(数字信号处理 fft).zip是一个压缩文件,它包含了用C语言编写的实现任意点数的FFT(快速傅里叶变换)和IFFT(逆快速傅里叶变换)算法的代码FFT和IFFT数字信号处理中常用的算法,用于将信号从时域变换到频域,或者从频域变换到时域。这些算法被广泛应用于信号处理、图像处理、通信系统等领域。 这个压缩文件中的代码可以实现任意点数的FFT和IFFT,也就是说可以处理不限于2的幂次的输入数据。一些常见库或者工具只支持2的幂次的FFT,而这个代码可以处理更为一般的情况。 通过使用这个代码,用户可以自定义输入数据的点数,并得到对应的FFT和IFFT结果。这对于一些特定的应用场景非常有用,比如需要处理采样率不同的信号,或者需要处理非常大的信号。 这个压缩文件中的代码是用C语言编写的,因此可以在大多数平台上运行。用户可以解压缩压缩文件,并将代码导入到自己的项目中进行使用。为了使用这些代码,用户需要了解FFT和IFFT的基本原理,并按照代码中的接口进行调用。 总之,这个压缩文件提供了实现任意点数的FFT和IFFT算法的代码,用户可以使用这个代码进行数字信号处理相关的工作。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

菜yuan~

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值