快速傅里叶变换-通过python代码实战讲解

  • **FFT介绍**
  • **适用范围**
  • **通过python代码来理解FFT**
  • **我们现在观察一下xf和xfp的值**
  • **采样定理**
  • **参考资料**



注:该文适合在jupyter notebook编辑器上演示

FFT介绍

FT、DFT、FFT的关系:
傅里叶变换(FT)是针对连续信号的,不适用于离散信号。而实际测得的信号(如振动、电流等)都是不连续即离散的,这个时候离散傅里叶变换(DFT)出现了,用来针对离散信号的做傅里叶变换。但是DFT从数学上计算是可行的,但给计算机计算,比较复杂,因此J.W.库利和T.W.图基提出了快速傅里叶变换 (FFT), 其目的是利用计算机来高效、快速计算离散傅里叶变换。

适用范围

周期性信号,平稳信号(简单可理解其频率不随时间变化)

通过python代码来理解FFT

先产生一段正弦信号 y=sin(2πf*t) ,这里信号频率f=2

import numpy as np # 导入numpy包
import matplotlib.pyplot as plt  # 导入matplotlib.pyplot包
t = np.linspace(start=0, stop=10, num=100)  # 采样频率fs=100÷10=10
y = 4*np.sin(2*np.pi*t*2) # y=sin(2*π*f*t) # 这里f=2
plt.plot(t, y)  # 绘制图形
plt.xlabel('t(s)')
plt.ylabel('f(Hz)')
plt.title('y=sin(2*π*2*t)')

输出结果

java 傅里叶 变换 傅里叶变换python代码_开发语言


这里有两个问题:

第一,纵坐标的-1,-2,-3和-4的负号未能显示完全,这是因为python不能显示负号和中文。

第二,这个正弦信号不光滑,比较尖锐,这是因为采样频率为10,分辨率比较低。

改进后的时域信号

import numpy as np # 导入numpy包
import matplotlib.pyplot as plt  # 导入matplotlib.pyplot包
import matplotlib
t = np.linspace(start=0, stop=10, num=1000)  # 采样频率fs=1000÷10=10
y = 4*np.sin(2*np.pi*t*2) # y=sin(2*π*f*t) # 这里f=2
matplotlib.rcParams['axes.unicode_minus']=False  #用来正常显示正负号
plt.plot(t, y)  # 绘制图形
plt.xlabel('t(s)') 
plt.ylabel('f(Hz)')
plt.title('y=sin(2*π*2*t)')

输出结果

java 傅里叶 变换 傅里叶变换python代码_python_02


现在采样频率为100,生成的信号和正弦信号很像了

现在就开始对这个离散信号做FFT变换,看看效果

####----进行fft---####
matplotlib.rcParams['axes.unicode_minus']=False  #用来正常显示正负号
t = np.linspace(start=0, stop=10, num=1000)  # 采样频率fs=1000÷10=10
y = 4*np.sin(2*np.pi*t*2) # y=sin(2*π*f*t) # 这里f=2
# 画时域图
ax1 = plt.subplot(1, 2, 1)
plt.plot(t, y)  
plt.xlabel('t(s)') 
plt.ylabel('f(Hz)')
plt.title('y=sin(2*π*2*t) 时域图')

# 画fft图
fs = 100   # 采样频率
xf = np.fft.fft(y)   # 对离散数据y做fft变换得到变换之后的数据xf
xfp = np.fft.fftfreq(len(y), d=1 / fs)  # fftfreq(length,fs),计算得到频率
xf = abs(xf) # 将复数求模,得到fft的幅值
ax1 = plt.subplot(1, 2, 2) 
plt.plot(xfp, xf) # 画fft图像,横坐标为频率,纵坐标为幅值
plt.xlabel('f(Hz)') 
plt.ylabel('Amp')
plt.title('y=sin(2*π*2*t)p fft图')

输出结果

java 傅里叶 变换 傅里叶变换python代码_矩阵_03


从fft图中可以看出这两个重要的现象

第一,频率存在负数,且其图像关于0Hz对称。在-2Hz和2Hz的地方存在很大的幅值

第二,频率是从-50Hz到50Hz(这个也是很重要的一点,fft得到的频率是采样频率的一半)

我们现在观察一下xf和xfp的值

print(xf)  #这是幅值
array([1.91846539e-13, 2.00035896e-01, 4.03098512e-01, 6.12369070e-01,
       8.31354662e-01, 1.06409657e+00, 1.31544309e+00, 1.59142838e+00,
       1.89982442e+00, 2.25098029e+00, 2.65915286e+00, 3.14471294e+00,
       3.73799357e+00, 4.48641986e+00, 5.46873781e+00, 6.82624293e+00,
       8.84061181e+00, 1.21663574e+01, 1.87537300e+01, 3.82003257e+01,
       1.99767920e+03, 4.18093223e+01, 2.11642707e+01, 1.43606392e+01,
       1.09658841e+01, 8.92699589e+00, 7.56418128e+00, 6.58718456e+00,
       5.85121855e+00, 5.27597074e+00, 4.81330295e+00, 4.43260128e+00,
       4.11346328e+00, 3.84176526e+00, 3.60741741e+00, 3.40301801e+00,
       ...
       1.99767920e+03, 3.82003257e+01, 1.87537300e+01, 1.21663574e+01,
       8.84061181e+00, 6.82624293e+00, 5.46873781e+00, 4.48641986e+00,
       3.73799357e+00, 3.14471294e+00, 2.65915286e+00, 2.25098029e+00,
       1.89982442e+00, 1.59142838e+00, 1.31544309e+00, 1.06409657e+00,
       8.31354662e-01, 6.12369070e-01, 4.03098512e-01, 2.00035896e-01])
print(xfp) #这是频率值
array([  0. ,   0.1,   0.2,   0.3,   0.4,   0.5,   0.6,   0.7,   0.8,
         0.9,   1. ,   1.1,   1.2,   1.3,   1.4,   1.5,   1.6,   1.7,
         1.8,   1.9,   2. ,   2.1,   2.2,   2.3,   2.4,   2.5,   2.6,
         2.7,   2.8,   2.9,   3. ,   3.1,   3.2,   3.3,   3.4,   3.5,
         3.6,   3.7,   3.8,   3.9,   4. ,   4.1,   4.2,   4.3,   4.4,
         4.5,   4.6,   4.7,   4.8,   4.9,   5. ,   5.1,   5.2,   5.3,
		...
		47.7,  47.8,  47.9,  48. ,  48.1,  48.2,  48.3,  48.4,  48.5,
        48.6,  48.7,  48.8,  48.9,  49. ,  49.1,  49.2,  49.3,  49.4,
        49.5,  49.6,  49.7,  49.8,  49.9, -50. , -49.9, -49.8, -49.7,
       -49.6, -49.5, -49.4, -49.3, -49.2, -49.1, -49. , -48.9, -48.8,
       -48.7, -48.6, -48.5, -48.4, -48.3, -48.2, -48.1, -48. , -47.9,
       -47.8, -47.7, -47.6, -47.5, -47.4, -47.3, -47.2, -47.1, -47. ,
       -46.9, -46.8, -46.7, -46.6, -46.5, -46.4, -46.3, -46.2, -46.1,
       ...
		-3.7,  -3.6,  -3.5,  -3.4,  -3.3,  -3.2,  -3.1,  -3. ,  -2.9,
        -2.8,  -2.7,  -2.6,  -2.5,  -2.4,  -2.3,  -2.2,  -2.1,  -2. ,
        -1.9,  -1.8,  -1.7,  -1.6,  -1.5,  -1.4,  -1.3,  -1.2,  -1.1,
        -1. ,  -0.9,  -0.8,  -0.7,  -0.6,  -0.5,  -0.4,  -0.3,  -0.2,
        -0.1])

从其频率值xfp可以看出来,其变化是0Hz->50Hz->-50Hz->0Hz

那我们只要0-50Hz的频率该怎么做,有两种办法
方法1
通过plt.xlim(x1, x2) #x1, x2是要展示出来的坐标值

ax1 = plt.subplot(1, 2, 2) 
plt.plot(xfp, xf) # 画fft图像
plt.xlabel('f(Hz)') 
plt.ylabel('Amp')
plt.title('y=sin(2*π*2*t)p fft图')
plt.xlim(0, int(fs/2))  #展示频率在(0,int(fs/2))范围内的图像

输出结果

java 傅里叶 变换 傅里叶变换python代码_java 傅里叶 变换_04


方法2

只取正频率,即取前半个长度的点数

xf = np.fft.fft(y)   # 对离散数据y做fft变换得到变换之后的数据xf
xf = abs(xf) # 将复数求模
xf = xf[0:int(len(xf)/2)]  #取xf前半个长度
xfp = np.fft.fftfreq(len(y), d=1 / fs)  # fftfreq(length,fs),计算得到频率
xfp = xfp[0:int(len(xfp)/2)]  #取xfp前半个长度
ax1 = plt.subplot(1, 2, 2) 
plt.plot(xfp, xf) # 画fft图像
plt.xlabel('f(Hz)') 
plt.ylabel('Amp')
plt.title('y=sin(2*π*2*t)p fft图')

输出结果

java 傅里叶 变换 傅里叶变换python代码_采样频率_05

采样定理

还记得第一张图,采样频率为10Hz时,其形状比较尖锐,那做fft是否也能分析出信号频率呢?
答案是可以的。采样定理是这样说的:采样频率需至少为所关心频率的2倍时,才能保留所关心信号的信息。采样10Hz是信号频率2Hz的5倍,大于2倍,所以是可以分析出来的

现在对采样频率为10Hz在的信号进行fft分析

####----进行fft---####
matplotlib.rcParams['axes.unicode_minus']=False  #用来正常显示正负号
t = np.linspace(start=0, stop=10, num=100)  # 采样频率fs=1000÷10=10
y = 4*np.sin(2*np.pi*t*2) # y=sin(2*π*f*t) # 这里f=2
# 画时域图
ax1 = plt.subplot(1, 2, 1)
plt.plot(t, y)  
plt.xlabel('t(s)') 
plt.ylabel('f(Hz)')
plt.title('y=sin(2*π*2*t) 时域图')

# 画fft图
fs = 10   # 采样频率
xf = np.fft.fft(y)   # 对离散数据y做fft变换得到变换之后的数据xf
xfp = np.fft.fftfreq(len(y), d=1 / fs)  # fftfreq(length,fs),计算得到频率
xf = abs(xf) # 将复数求模
ax1 = plt.subplot(1, 2, 2) 
plt.plot(xfp, xf) # 画fft图像
plt.xlim(0, int(fs/2))
plt.xlabel('f(Hz)') 
plt.ylabel('Amp')
plt.title('y=sin(2*π*2*t)p fft图')

结果

java 傅里叶 变换 傅里叶变换python代码_java 傅里叶 变换_06


由图可知,当采样频率比较低时,其在频率为2Hz附近也会存在幅值比较低的频率,这种现象叫频谱泄露。

参考资料

信号处理 | 傅里叶变换、短时傅里叶变换、小波变换、希尔伯特变换、希尔伯特黄变换DSP基本名词术语及其关系信号处理若干名词解释(I)信号处理若干名词解释(II)什么是泄漏?什么是混叠什么是窗函数