傅里叶变换算法

[复制链接]
3431|34
 楼主| 沉浮的茶叶 发表于 2022-5-2 13:10 | 显示全部楼层 |阅读模式
摘要:傅里叶变换的核心在于,“任何连续周期信号可以由一组适当的正弦曲线组合而成”,在这个基础上对信号的中特定频率的正弦波进行分解或者重组,基于频率方面分析波形。
原文出处:https://mp.weixin.qq.com/s/5VIpNWci9YLY3m4gcYd6-w
 楼主| 沉浮的茶叶 发表于 2022-5-2 13:11 | 显示全部楼层
1、傅里叶变换的意义

近似周期性的方波(橙色),可采用6组正弦波(蓝色)合成,这是傅里叶的基础。

85899626f67e59c535.png

对数字信号处理或者工程数学有一定基础,就明白傅里叶变换的价值。一般情况下的信号或者波形随时间变化,称为时域信号,时域(Time domain)是描述数学函数或物理信号对时间的关系。而以频率和幅度表示信号称为频域,频域(frequency domain)是描述信号在频率方面特性时用到的一种坐标系。其数学理论上暂且不理,针对嵌入式系统开发,只探讨其物理意义或者应用场景。

时域和频域是信号的基本性质,时域的表示较为形象与直观,比较符合一般认知,而频域分析则更为简练,剖析问题更为深刻和方便。

例如下图的左侧时域信号,其可以分解为2路正弦波的叠加效果,右侧为频域信号,表示2路正弦波的频率和幅度。傅里叶变化可简单理解为求解一段信号或波形,由哪些正弦波组成,也可以反向推导多路正弦波合并后的效果。

18865626f67f19112f.png

傅里叶变换是一种信号分析方法,让我们对信号的构成和特点进行深入的、定量的研究。把信号通过频谱的方式进行准确的、定量的描述。将原来难以处理的时域信号转换成了易于分析的频域信号,即傅里叶变换的核心是从时域到频域的变换。


 楼主| 沉浮的茶叶 发表于 2022-5-2 13:12 | 显示全部楼层
2、变换方式

数字信号属于离散值,对应的称为离散傅里叶变换(DFT),是傅里叶变换在时域和频域上都呈现离散的形式,将时域信号的采样变换为在离散时间傅里叶变换(DTFT)频域的采样。在形式上,变换两端(时域和频域上)的序列是有限长的,而实际上这两组序列都应当被认为是离散周期信号的主值序列。即使对有限长的离散信号作DFT,也应当将其看作经过周期延拓成为周期信号再作变换。

在实际应用中通常采用快速傅里叶变换以高效计算效率,快速傅里叶变换 FFT(Fast Fourier Transformation)是离散傅里叶变换 DFT(Discrete Fourier Transform)的快速算法。采用这种算法能使计算机计算离散傅里叶变换所需要的乘法次数大为减少,特别是被变换的抽样点数N越多,FFT算法计算量的节省就越显著。同理,从频域到时域的变换,称为逆变换,快速傅里叶逆变换 IFFT和离散傅里叶逆变换IDFT。


 楼主| 沉浮的茶叶 发表于 2022-5-2 13:12 | 显示全部楼层
3、应用

一般嵌入式系统使用快速傅里叶变换是分析某段信号中混合的杂波干扰,或者剔除某个频段后再逆变换。有些高端示波器可以对信号快速傅里叶变换 FFT,直接显示高频杂波的频率或者较大幅度干扰源的频率,以便由针对性的检查电路窜扰。本文主要是讲解使用fft/ifft进行干扰信号的过滤。

1、python生成正弦波,以及混合波形,提取数值作为c语言FFT/IFFT的数据源

2、先进行FFT,输出幅频数据,导入Excel看看效果

3、对数据进行简单过滤

4、过滤后的数据进行IFFT运算,再导入Excel看还原的效果


 楼主| 沉浮的茶叶 发表于 2022-5-2 13:13 | 显示全部楼层
3.1 基于python生成数据源
  1. # This is a Python script.

  2. import numpy as np
  3. import matplotlib.pyplot as plt
  4. def sin_wave(A, f, fs, phi, t):
  5.     '''
  6.     :params A:    振幅
  7.     :params f:    信号频率
  8.     :params fs:   采样频率
  9.     :params phi:  相位
  10.     :params t:    时间长度
  11.     '''
  12.     # 若时间序列长度为 t=1s,
  13.     # 采样频率 fs=1000 Hz, 则采样时间间隔 Ts=1/fs=0.001s
  14.     # 采样点个数为 n=t/Ts=1/0.001=1000, 即有1000个点,每个点间隔为 Ts
  15.     Ts = 1/fs
  16.     n = t / Ts
  17.     n = np.arange(n)
  18.     y = A*np.sin(2*np.pi*f*n*Ts + phi*(np.pi/180))
  19.     return y

  20. fs = 360*40
  21. my_sin1 = sin_wave(100, 100, fs=fs, phi=0, t=0.08)
  22. my_sin2 = sin_wave(20, 500, fs=fs, phi=0, t=0.08)
  23. my_sin3 = sin_wave(10, 1100, fs=fs, phi=0, t=0.08)

  24. x = np.arange(0, 0.08, 1/fs)
  25. plt.xlabel('x-t (samRate=14400)')
  26. plt.ylabel('y-A')
  27. plt.grid()
  28. plt.plot(x, my_sin1, 'k--',label="f=100")
  29. plt.plot(x, my_sin2, 'b--',label="f=500")
  30. plt.plot(x, my_sin3, 'g--',label="f=1100")
  31. plt.plot(x, my_sin1+my_sin2+my_sin3, 'r',label="mix")
  32. plt.legend()
  33. plt.show()

  34. np.savetxt("sin.txt",my_sin1+my_sin2+my_sin3, fmt='%.06f')

  35. def print_name(name):
  36.     print(f'Hi, {name}')


  37. if __name__ == '__main__':
  38.     print_name('fft test')
生成的波形图如下:
4578626f68720a035.png
三种频率的sin以及合成后的红色曲线,变形的正弦波;同时也将数据存入文件sin.txt备用。

 楼主| 沉浮的茶叶 发表于 2022-5-2 13:14 | 显示全部楼层
3.2 基于C验证算法

将前面生成的数据使用c语言的FFT/IFFT进行处理,输出结果,并导入Excels生成图表展示效果。

  1. #include <math.h>
  2. #include <stdio.h>
  3. #include "string.h"

  4. typedef struct
  5. {
  6.     float real;
  7.     float imag;
  8. } complex_t;

  9. #ifndef PI
  10. #define PI             (3.14159265)
  11. #endif

  12. #define TYPE_FFT_E     float    /* Type is the same with complex_t member */

  13. typedef complex_t TYPE_FFT;  /* Define complex_t in Config.h */
  14. typedef unsigned int                      uint32_t;

  15. #define     SAMPLE_NODES    (1024)
  16. complex_t   fft_buff[SAMPLE_NODES];

  17. //python生成的3个sin混合的波形数组
  18. float my_sin_wave_table[] =
  19. {
  20.     0.000000,13.308217,25.359460,35.142296,42.082633,46.140080,47.788611,47.887148,47.470293,47.507139,48.682920,51.252751,\
  21.     55.000000,59.307883,63.326941,66.199152,67.286436,66.350198,63.639610,59.866963,56.074110,53.418823,52.928233,55.274323,\
  22.     60.621778,68.582540,78.287693,88.561119,98.156743,106.007043,111.428222,114.237346,114.756729,113.706291,112.009861,\
  23.     110.560634,110.000000,110.560634,112.009861,113.706291,114.756729,114.237346,111.428222,106.007043,98.156743,88.561119,\
  24.     78.287693,68.582540,60.621778,55.274323,52.928233,53.418823,56.074110,59.866963,63.639610,66.350198,67.286436,66.199152,\
  25.     63.326941,59.307883,55.000000,51.252751,48.682920,47.507139,47.470293,47.887148,47.788611,46.140080,42.082633,35.142296,\
  26.     25.359460,13.308217,0.000000,-13.308217,-25.359460,-35.142296,-42.082633,-46.140080,-47.788611,-47.887148,-47.470293,\
  27.     -47.507139,-48.682920,-51.252751,-55.000000,-59.307883,-63.326941,-66.199152,-67.286436,-66.350198,-63.639610,-59.866963,\
  28.     -56.074110,-53.418823,-52.928233,-55.274323,-60.621778,-68.582540,-78.287693,-88.561119,-98.156743,-106.007043,-111.428222,\
  29.     -114.237346,-114.756729,-113.706291,-112.009861,-110.560634,-110.000000,-110.560634,-112.009861,-113.706291,-114.756729,\
  30.     -114.237346,-111.428222,-106.007043,-98.156743,-88.561119,-78.287693,-68.582540,-60.621778,-55.274323,-52.928233,\
  31.     -53.418823,-56.074110,-59.866963,-63.639610,-66.350198,-67.286436,-66.199152,-63.326941,-59.307883,-55.000000,\
  32.     -51.252751,-48.682920,-47.507139,-47.470293,-47.887148,-47.788611,-46.140080,-42.082633,-35.142296,-25.359460,\
  33.     -13.308217,-0.000000,13.308217,25.359460,35.142296,42.082633,46.140080,47.788611,47.887148,47.470293,47.507139,\
  34.     48.682920,51.252751,55.000000,59.307883,63.326941,66.199152,67.286436,66.350198,63.639610,59.866963,56.074110,\
  35.     53.418823,52.928233,55.274323,60.621778,68.582540,78.287693,88.561119,98.156743,106.007043,111.428222,114.237346,\
  36.     114.756729,113.706291,112.009861,110.560634,110.000000,110.560634,112.009861,113.706291,114.756729,114.237346,\
  37.     111.428222,106.007043,98.156743,88.561119,78.287693,68.582540,60.621778,55.274323,52.928233,53.418823,56.074110,\
  38.     59.866963,63.639610,66.350198,67.286436,66.199152,63.326941,59.307883,55.000000,51.252751,48.682920,47.507139,\
  39.     47.470293,47.887148,47.788611,46.140080,42.082633,35.142296,25.359460,13.308217,0.000000,-13.308217,-25.359460,\
  40.     -35.142296,-42.082633,-46.140080,-47.788611,-47.887148,-47.470293,-47.507139,-48.682920,-51.252751,-55.000000,\
  41.     -59.307883,-63.326941,-66.199152,-67.286436,-66.350198,-63.639610,-59.866963,-56.074110,-53.418823,-52.928233,\
  42.     -55.274323,-60.621778,-68.582540,-78.287693,-88.561119,-98.156743,-106.007043,-111.428222,-114.237346,-114.756729,\
  43.     -113.706291,-112.009861,-110.560634,-110.000000,-110.560634,-112.009861,-113.706291,-114.756729,-114.237346,-111.428222,\
  44.     -106.007043,-98.156743,-88.561119,-78.287693,-68.582540,-60.621778,-55.274323,-52.928233,-53.418823,-56.074110,-59.866963,\
  45.     -63.639610,-66.350198,-67.286436,-66.199152,-63.326941,-59.307883,-55.000000,-51.252751,-48.682920,-47.507139,-47.470293,\
  46.     -47.887148,-47.788611,-46.140080,-42.082633,-35.142296,-25.359460,-13.308217,-0.000000,13.308217,25.359460,35.142296,\
  47.     42.082633,46.140080,47.788611,47.887148,47.470293,47.507139,48.682920,51.252751,55.000000,59.307883,63.326941,66.199152,\
  48.     67.286436,66.350198,63.639610,59.866963,56.074110,53.418823,52.928233,55.274323,60.621778,68.582540,78.287693,88.561119,\
  49.     98.156743,106.007043,111.428222,114.237346,114.756729,113.706291,112.009861,110.560634,110.000000,110.560634,112.009861,\
  50.     113.706291,114.756729,114.237346,111.428222,106.007043,98.156743,88.561119,78.287693,68.582540,60.621778,55.274323,\
  51.     52.928233,53.418823,56.074110,59.866963,63.639610,66.350198,67.286436,66.199152,63.326941,59.307883,55.000000,51.252751,\
  52.     48.682920,47.507139,47.470293,47.887148,47.788611,46.140080,42.082633,35.142296,25.359460,13.308217,0.000000,-13.308217,\
  53.     -25.359460,-35.142296,-42.082633,-46.140080,-47.788611,-47.887148,-47.470293,-47.507139,-48.682920,-51.252751,-55.000000,\
  54.     -59.307883,-63.326941,-66.199152,-67.286436,-66.350198,-63.639610,-59.866963,-56.074110,-53.418823,-52.928233,-55.274323,\
  55.     -60.621778,-68.582540,-78.287693,-88.561119,-98.156743,-106.007043,-111.428222,-114.237346,-114.756729,-113.706291,\
  56.     -112.009861,-110.560634,-110.000000,-110.560634,-112.009861,-113.706291,-114.756729,-114.237346,-111.428222,-106.007043,\
  57.     -98.156743,-88.561119,-78.287693,-68.582540,-60.621778,-55.274323,-52.928233,-53.418823,-56.074110,-59.866963,-63.639610,\
  58.     -66.350198,-67.286436,-66.199152,-63.326941,-59.307883,-55.000000,-51.252751,-48.682920,-47.507139,-47.470293,-47.887148,\
  59.     -47.788611,-46.140080,-42.082633,-35.142296,-25.359460,-13.308217,-0.000000,13.308217,25.359460,35.142296,42.082633,\
  60.     46.140080,47.788611,47.887148,47.470293,47.507139,48.682920,51.252751,55.000000,59.307883,63.326941,66.199152,67.286436,\
  61.     66.350198,63.639610,59.866963,56.074110,53.418823,52.928233,55.274323,60.621778,68.582540,78.287693,88.561119,98.156743,\
  62.     106.007043,111.428222,114.237346,114.756729,113.706291,112.009861,110.560634,110.000000,110.560634,112.009861,113.706291,\
  63.     114.756729,114.237346,111.428222,106.007043,98.156743,88.561119,78.287693,68.582540,60.621778,55.274323,52.928233,53.418823,\
  64.     56.074110,59.866963,63.639610,66.350198,67.286436,66.199152,63.326941,59.307883,55.000000,51.252751,48.682920,47.507139,\
  65.     47.470293,47.887148,47.788611,46.140080,42.082633,35.142296,25.359460,13.308217,0.000000,-13.308217,-25.359460,-35.142296,\
  66.     -42.082633,-46.140080,-47.788611,-47.887148,-47.470293,-47.507139,-48.682920,-51.252751,-55.000000,-59.307883,-63.326941,\
  67.     -66.199152,-67.286436,-66.350198,-63.639610,-59.866963,-56.074110,-53.418823,-52.928233,-55.274323,-60.621778,-68.582540,\
  68.     -78.287693,-88.561119,-98.156743,-106.007043,-111.428222,-114.237346,-114.756729,-113.706291,-112.009861,-110.560634,\
  69.     -110.000000,-110.560634,-112.009861,-113.706291,-114.756729,-114.237346,-111.428222,-106.007043,-98.156743,-88.561119,\
  70.     -78.287693,-68.582540,-60.621778,-55.274323,-52.928233,-53.418823,-56.074110,-59.866963,-63.639610,-66.350198,-67.286436,\
  71.     -66.199152,-63.326941,-59.307883,-55.000000,-51.252751,-48.682920,-47.507139,-47.470293,-47.887148,-47.788611,-46.140080,\
  72.     -42.082633,-35.142296,-25.359460,-13.308217,-0.000000,13.308217,25.359460,35.142296,42.082633,46.140080,47.788611,47.887148,\
  73.     47.470293,47.507139,48.682920,51.252751,55.000000,59.307883,63.326941,66.199152,67.286436,66.350198,63.639610,59.866963,\
  74.     56.074110,53.418823,52.928233,55.274323,60.621778,68.582540,78.287693,88.561119,98.156743,106.007043,111.428222,114.237346,\
  75.     114.756729,113.706291,112.009861,110.560634,110.000000,110.560634,112.009861,113.706291,114.756729,114.237346,111.428222,\
  76.     106.007043,98.156743,88.561119,78.287693,68.582540,60.621778,55.274323,52.928233,53.418823,56.074110,59.866963,63.639610,\
  77.     66.350198,67.286436,66.199152,63.326941,59.307883,55.000000,51.252751,48.682920,47.507139,47.470293,47.887148,47.788611,\
  78.     46.140080,42.082633,35.142296,25.359460,13.308217,0.000000,-13.308217,-25.359460,-35.142296,-42.082633,-46.140080,-47.788611,\
  79.     -47.887148,-47.470293,-47.507139,-48.682920,-51.252751,-55.000000,-59.307883,-63.326941,-66.199152,-67.286436,-66.350198,\
  80.     -63.639610,-59.866963,-56.074110,-53.418823,-52.928233,-55.274323,-60.621778,-68.582540,-78.287693,-88.561119,-98.156743,\
  81.     -106.007043,-111.428222,-114.237346,-114.756729,-113.706291,-112.009861,-110.560634,-110.000000,-110.560634,-112.009861,\
  82.     -113.706291,-114.756729,-114.237346,-111.428222,-106.007043,-98.156743,-88.561119,-78.287693,-68.582540,-60.621778,-55.274323,\
  83.     -52.928233,-53.418823,-56.074110,-59.866963,-63.639610,-66.350198,-67.286436,-66.199152,-63.326941,-59.307883,-55.000000,\
  84.     -51.252751,-48.682920,-47.507139,-47.470293,-47.887148,-47.788611,-46.140080,-42.082633,-35.142296,-25.359460,-13.308217,\
  85.     -0.000000,13.308217,25.359460,35.142296,42.082633,46.140080,47.788611,47.887148,47.470293,47.507139,48.682920,51.252751,\
  86.     55.000000,59.307883,63.326941,66.199152,67.286436,66.350198,63.639610,59.866963,56.074110,53.418823,52.928233,55.274323,\
  87.     60.621778,68.582540,78.287693,88.561119,98.156743,106.007043,111.428222,114.237346,114.756729,113.706291,112.009861,\
  88.     110.560634,110.000000,110.560634,112.009861,113.706291,114.756729,114.237346,111.428222,106.007043,98.156743,88.561119,\
  89.     78.287693,68.582540,60.621778,55.274323,52.928233,53.418823,56.074110,59.866963,63.639610,66.350198,67.286436,66.199152,\
  90.     63.326941,59.307883,55.000000,51.252751,48.682920,47.507139,47.470293,47.887148,47.788611,46.140080,42.082633,35.142296,\
  91.     25.359460,13.308217,0.000000,-13.308217,-25.359460,-35.142296,-42.082633,-46.140080,-47.788611,-47.887148,-47.470293,\
  92.     -47.507139,-48.682920,-51.252751,-55.000000,-59.307883,-63.326941,-66.199152,-67.286436,-66.350198,-63.639610,-59.866963,\
  93.     -56.074110,-53.418823,-52.928233,-55.274323,-60.621778,-68.582540,-78.287693,-88.561119,-98.156743,-106.007043,-111.428222,\
  94.     -114.237346,-114.756729,-113.706291,-112.009861,-110.560634,-110.000000,-110.560634,-112.009861,-113.706291,-114.756729,\
  95.     -114.237346,-111.428222,-106.007043,-98.156743,-88.561119,-78.287693,-68.582540,-60.621778,-55.274323,-52.928233,-53.418823,\
  96.     -56.074110,-59.866963,-63.639610,-66.350198,-67.286436,-66.199152,-63.326941,-59.307883,-55.000000,-51.252751,-48.682920,\
  97.     -47.507139,-47.470293,-47.887148,-47.788611,-46.140080,-42.082633,-35.142296,-25.359460,-13.308217,-0.000000,13.308217,\
  98.     25.359460,35.142296,42.082633,46.140080,47.788611,47.887148,47.470293,47.507139,48.682920,51.252751,55.000000,59.307883,\
  99.     63.326941,66.199152,67.286436,66.350198,63.639610,59.866963,56.074110,53.418823,52.928233,55.274323,60.621778,68.582540,\
  100.     78.287693,88.561119,98.156743,106.007043,111.428222,114.237346,114.756729,113.706291,112.009861,110.560634,110.000000,\
  101.     110.560634,112.009861,113.706291,114.756729,114.237346,111.428222,106.007043,98.156743,88.561119,78.287693,68.582540,\
  102.     60.621778,55.274323,52.928233,53.418823,56.074110,59.866963,63.639610,66.350198,67.286436,66.199152,63.326941,59.307883,\
  103.     55.000000,51.252751,48.682920,47.507139,47.470293,47.887148,47.788611,46.140080,42.082633,35.142296,25.359460,13.308217,\
  104.     0.000000,-13.308217,-25.359460,-35.142296,-42.082633,-46.140080,-47.788611,-47.887148,-47.470293,-47.507139,-48.682920,\
  105.     -51.252751,-55.000000,-59.307883,-63.326941,-66.199152,-67.286436,-66.350198,-63.639610,-59.866963,-56.074110,-53.418823,\
  106.     -52.928233,-55.274323,-60.621778,-68.582540,-78.287693,-88.561119,-98.156743,-106.007043,-111.428222,-114.237346,-114.756729,\
  107.     -113.706291,-112.009861,-110.560634,-110.000000,-110.560634,-112.009861,-113.706291,-114.756729,-114.237346,-111.428222,\
  108.     -106.007043,-98.156743,-88.561119,-78.287693,-68.582540,-60.621778,-55.274323,-52.928233,-53.418823,-56.074110,-59.866963,\
  109.     -63.639610,-66.350198,-67.286436,-66.199152,-63.326941,-59.307883,-55.000000,-51.252751,-48.682920,-47.507139,-47.470293,\
  110.     -47.887148,-47.788611,-46.140080,-42.082633,-35.142296,-25.359460,-13.308217,-0.000000,13.308217,25.359460,35.142296,\
  111.     42.082633,46.140080,47.788611,47.887148,47.470293,47.507139,48.682920,51.252751,55.000000,59.307883,63.326941,66.199152,\
  112.     67.286436,66.350198,63.639610,59.866963,56.074110,53.418823,52.928233,55.274323,60.621778,68.582540,78.287693,88.561119,\
  113.     98.156743,106.007043,111.428222,114.237346,114.756729,113.706291,112.009861,110.560634,110.000000,110.560634,112.009861,\
  114.     113.706291,114.756729,114.237346,111.428222,106.007043,98.156743,88.561119,78.287693,68.582540,60.621778,55.274323,52.928233,\
  115.     53.418823,56.074110,59.866963,63.639610,66.350198,67.286436,66.199152,63.326941,59.307883,55.000000,51.252751,48.682920,\
  116.     47.507139,47.470293,47.887148,47.788611,46.140080,42.082633,35.142296,25.359460,13.308217,0.000000,-13.308217,-25.359460,\
  117.     -35.142296,-42.082633,-46.140080,-47.788611,-47.887148,-47.470293,-47.507139,-48.682920,-51.252751,-55.000000,-59.307883,\
  118.     -63.326941,-66.199152,-67.286436,-66.350198,-63.639610,-59.866963,-56.074110,-53.418823,-52.928233,-55.274323,-60.621778,\
  119.     -68.582540,-78.287693,-88.561119,-98.156743,-106.007043,-111.428222,-114.237346,-114.756729,-113.706291,-112.009861,\
  120.     -110.560634,-110.000000,-110.560634,-112.009861,-113.706291,-114.756729,-114.237346,-111.428222,-106.007043,-98.156743,\
  121.     -88.561119,-78.287693,-68.582540,-60.621778,-55.274323,-52.928233,-53.418823,-56.074110,-59.866963,-63.639610,-66.350198,\
  122.     -67.286436,-66.199152,-63.326941,-59.307883,-55.000000,-51.252751,-48.682920,-47.507139,-47.470293,-47.887148,-47.788611,\
  123.     -46.140080,-42.082633,-35.142296,-25.359460,-13.308217,
  124. };

  125. //fft算法来自开源 https://github.com/xiahouzuoxin/fft

  126. const float sin_tb[] =    // 精度(PI PI/2 PI/4 PI/8 PI/16 ... PI/(2^k))
  127. {
  128.         0.000000, 1.000000, 0.707107, 0.382683, 0.195090, 0.098017,
  129.         0.049068, 0.024541, 0.012272, 0.006136, 0.003068, 0.001534,
  130.         0.000767, 0.000383, 0.000192, 0.000096, 0.000048, 0.000024,
  131.         0.000012, 0.000006, 0.000003
  132.         };

  133. const float cos_tb[] =    // 精度(PI PI/2 PI/4 PI/8 PI/16 ... PI/(2^k))
  134. {
  135.         -1.000000, 0.000000, 0.707107, 0.923880, 0.980785, 0.995185,
  136.         0.998795, 0.999699, 0.999925, 0.999981, 0.999995, 0.999999,
  137.         1.000000, 1.000000, 1.000000, 1.000000, 1.000000, 1.000000,
  138.         1.000000, 1.000000, 1.000000
  139.         };


  140. int ones_32(uint32_t n)
  141. {
  142.     unsigned int c = 0 ;
  143.     for(c = 0; n; ++c)
  144.     {
  145.         n &= (n - 1) ;
  146.     }

  147.     return c ;
  148. }


  149. uint32_t floor_log2_32(uint32_t fft_buff)
  150. {
  151.     fft_buff |= (fft_buff >> 1);
  152.     fft_buff |= (fft_buff >> 2);
  153.     fft_buff |= (fft_buff >> 4);
  154.     fft_buff |= (fft_buff >> 8);
  155.     fft_buff |= (fft_buff >> 16);

  156.     return (ones_32(fft_buff >> 1));
  157. }


  158. /*
  159. * FFT Algorithm
  160. * === Inputs ===
  161. * fft_buff : complex numbers
  162. * N : nodes of FFT. @N should be power of 2, that is 2^(*)
  163. * === Output ===
  164. * the @fft_buff contains the result of FFT algorithm, so the original data
  165. * in @fft_buff is destroyed, please store them before using FFT.
  166. */
  167. int fft(TYPE_FFT *fft_buff, uint32_t N)
  168. {
  169.     int i, j, l, k, ip;
  170.     static uint32_t M = 0;
  171.     static int le, le2;
  172.     static TYPE_FFT_E sR, sI, tR, tI, uR, uI;

  173.     M = floor_log2_32(N);

  174.     /*
  175.      * bit reversal sorting
  176.      */

  177.     l = N >> 1;
  178.     j = l;
  179.     ip = N - 2;
  180.     for(i = 1; i <= ip; i++)
  181.     {
  182.         if(i < j)
  183.         {
  184.             tR = fft_buff[j].real;
  185.             tI = fft_buff[j].imag;
  186.             fft_buff[j].real = fft_buff[i].real;
  187.             fft_buff[j].imag = fft_buff[i].imag;
  188.             fft_buff[i].real = tR;
  189.             fft_buff[i].imag = tI;
  190.         }

  191.         k = l;

  192.         while(k <= j)
  193.         {
  194.             j = j - k;
  195.             k = k >> 1;
  196.         }

  197.         j = j + k;
  198.     }

  199.     /*
  200.      * For Loops
  201.      */

  202.     for(l = 1; l <= M; l++)  /* loop for ceil{log2(N)} */
  203.     {
  204.         le  = (int)(1 << l);
  205.         le2 = (int)(le >> 1);
  206.         uR = 1;
  207.         uI = 0;

  208.         k = floor_log2_32(le2);
  209.         sR = cos_tb[k];
  210.         sI = -sin_tb[k];

  211.         for(j = 1; j <= le2; j++)  /* loop for each sub DFT */
  212.         {
  213.             for(i = j - 1; i < N; i += le) /* loop for each butterfly */
  214.             {
  215.                 ip = i + le2;
  216.                 tR = fft_buff[ip].real * uR - fft_buff[ip].imag * uI;
  217.                 tI = fft_buff[ip].real * uI + fft_buff[ip].imag * uR;
  218.                 fft_buff[ip].real = fft_buff[i].real - tR;
  219.                 fft_buff[ip].imag = fft_buff[i].imag - tI;
  220.                 fft_buff[i].real += tR;
  221.                 fft_buff[i].imag += tI;
  222.             }  /* Next i */

  223.             tR = uR;
  224.             uR = tR * sR - uI * sI;
  225.             uI = tR * sI + uI * sR;

  226.         } /* Next j */

  227.     } /* Next l */

  228.     return 0;
  229. }


  230. /*
  231. * Inverse FFT Algorithm
  232. * === Inputs ===
  233. * fft_buff : complex numbers
  234. * N : nodes of FFT. @N should be power of 2, that is 2^(*)
  235. * === Output ===
  236. * the @fft_buff contains the result of FFT algorithm, so the original data
  237. * in @fft_buff is destroyed, please store them before using FFT.
  238. */

  239. int ifft(TYPE_FFT *fft_buff, uint32_t N)
  240. {
  241.     int k = 0;

  242.     for(k = 0; k <= N - 1; k++)
  243.     {
  244.         fft_buff[k].imag = -fft_buff[k].imag;
  245.     }

  246.     fft(fft_buff, N);    /* using FFT */

  247.     for(k = 0; k <= N - 1; k++)
  248.     {
  249.         fft_buff[k].real = fft_buff[k].real / N;
  250.         fft_buff[k].imag = -fft_buff[k].imag / N;
  251.     }

  252.     return 0;
  253. }

  254. static void import_data(void)
  255. {
  256.     int i;

  257.     for(i = 0; i < SAMPLE_NODES; i++)
  258.     {
  259.         fft_buff[i].real = my_sin_wave_table[i];//取前1024个数进行fft变换
  260.         fft_buff[i].imag  = 0.0f;
  261.     }
  262. }


  263. int main(int argc, char *argv[])
  264. {
  265.     int i;
  266.     int f;//频率
  267.     float a;//幅度
  268.     int fd;
  269.     float t;

  270.     printf("FFT\r\n");

  271.     import_data();

  272.     fft(fft_buff, SAMPLE_NODES);

  273.     //fft后的结果在fft_buff
  274.     //将其实部与虚部处理,输出频点与幅度值,导入Excel看效果
  275.     //因为是周期性质,取前半部分即可

  276.     //数据采样频率是360*40,均分到SAMPLE_NODES,则对应频点间隔是 360*40/SAMPLE_NODES
  277.     fd=360*40/SAMPLE_NODES;

  278.     for(i = 0; i < SAMPLE_NODES / 2; i++)
  279.     {
  280.         f = i *fd;
  281.         a = (double)sqrt(fft_buff[i].real * fft_buff[i].real + fft_buff[i].imag * fft_buff[i].imag)/ (SAMPLE_NODES / 2);//转换幅度
  282.         //printf("%d,%f\n", f, a);//>>导入excel查看幅频图效果
  283.     }

  284.     //过滤高频部分
  285.     //将幅度小于某个值的,以30为例过滤
  286.     for(i = 0; i < SAMPLE_NODES; i++)
  287.     {
  288.         a = (double)sqrt(fft_buff[i].real * fft_buff[i].real + fft_buff[i].imag * fft_buff[i].imag)/ (SAMPLE_NODES / 2);
  289.         if(a<30)
  290.         {
  291.             fft_buff[i].real = 0;
  292.             fft_buff[i].imag = 0;
  293.         }
  294.     }

  295.     //再进行逆运算还原
  296.     ifft(fft_buff, SAMPLE_NODES);
  297.     for(i = 0; i < SAMPLE_NODES; i++)
  298.     {
  299.         t=1.0/(360*40)*i;//结合采样频率步进,方便查看波形效果
  300.         //printf("%f,%f\n", t,fft_buff[i].real);//>>导入excel查看还原后的sin效果
  301.     }

  302.     return 0;
  303. }


 楼主| 沉浮的茶叶 发表于 2022-5-2 13:15 | 显示全部楼层
3.3 幅频图

混合波形使用FFT后的幅频图:

83184626f68c35758a.png

看图可知对应的频点和幅度,与原始混合波形的3种正弦波吻合。

栅栏效应:其中频点与采样频率和FFT转换的采样数有关,需要结合实际定义,因为数字离散信号,频点并不是连续的,如上大概相差14,因为数据点数N为1024点,原始信号的采样频率为fs=14400Hz,则频率间隔fs/N约等于14Hz。这样看频谱图只有fs/N倍数的,例如没有410Hz,只能看到相邻频率406Hz和420Hz处频谱泄漏的数据,相当于透过栅栏观赏风景,只能看到频谱的一部分,而其它频率点看不见,因此很可能部分有用的频率成分被漏掉,这种现象被称为栅栏效应。


 楼主| 沉浮的茶叶 发表于 2022-5-2 13:15 | 显示全部楼层
3.4 逆变换还原

FFT后的幅频数据过滤其中2个频点的小幅度波形,还原后的效果如下图:

78063626f68e85b5f8.png

过滤还原后的波形,与原始图,前面python生成的黑色虚线吻合。


 楼主| 沉浮的茶叶 发表于 2022-5-2 13:16 | 显示全部楼层
4、应用

傅里叶变换在数字信号领域中应用广泛,在嵌入式系统中,一般用来分析ADC高速采集的数据、滤波处理,或者对音频进行简单分析。但傅里叶变换算法内存消耗有点大,低端设备可能无法运行。


tpgf 发表于 2022-6-1 11:13 | 显示全部楼层
傅里叶变换是经典的
aoyi 发表于 2022-6-1 11:32 | 显示全部楼层
想要理解透彻也不容易啊
nawu 发表于 2022-6-1 11:53 | 显示全部楼层
这个怎么确定需要运行的最低条件呢
zljiu 发表于 2022-6-1 12:01 | 显示全部楼层
对频率的范围有要求吗
gwsan 发表于 2022-6-1 12:12 | 显示全部楼层
傅里叶对ad的最低速度有要求吗
tfqi 发表于 2022-6-1 12:19 | 显示全部楼层
是的 就是对资源消耗的比较多
foxsbig 发表于 2022-6-3 15:19 | 显示全部楼层
这个一直就没搞明白过
107485133 发表于 2022-7-25 17:32 | 显示全部楼层
傅里叶变换算法
pixhw 发表于 2022-12-14 11:16 | 显示全部楼层
求一个关于128点的快速傅立叶的C语言程序  
lihuami 发表于 2022-12-14 11:25 | 显示全部楼层
FFT的公式是什么和算法是怎样实现
beacherblack 发表于 2022-12-15 11:06 | 显示全部楼层
怎样用C语言实现FFT算法啊?
您需要登录后才可以回帖 登录 | 注册

本版积分规则

12

主题

94

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部