python傅⾥叶谐波分析_Python机器学习(五⼗
六)SciPyfftpack(傅⾥叶变换)
SciPy提供了fftpack模块,包含了傅⾥叶变换的算法实现。
傅⾥叶变换把信号从时域变换到频域,以便对信号进⾏处理。傅⾥叶变换在信号与噪声处理、图像处理、⾳频信号处理等领域得到了⼴泛应⽤。
如需进⼀步了解傅⾥叶变换原理,可以参考相关资料。
快速傅⾥叶变换
计算机只能处理离散信号,使⽤离散傅⾥叶变换(DFT) 是计算机分析信号的基本⽅法。但是离散傅⾥叶变换的缺点是:计算量⼤,时间复杂度太⾼,当采样点数太⾼的时候,计算缓慢,由此出现了DFT的快速实现,即快速傅⾥叶变换FFT。
快速傅⾥叶变换(FFT)是计算量更⼩的离散傅⾥叶变换的⼀种实现⽅法,其逆变换被称为快速傅⾥叶逆变换(IFFT)。
⽰例
先对数据进⾏fft变换,然后再ifft逆变换。
importnumpy as np#从fftpack中导⼊fft(快速傅⾥叶变化)和ifft(快速傅⾥叶逆变换)函数
from scipy.fftpack importfft,ifft#创建⼀个随机值数组
x = np.array([1.0, 2.0, 1.0, -1.0, 1.5])#对数组数据进⾏傅⾥叶变换
y =fft(x)print('fft:')print(y)print('\n')#快速傅⾥叶逆变换
yinv =ifft(y)print('ifft:')print(yinv)print('\n')
输出
fft:
index复数[4.5 +0.j 2.08155948-1.65109876j -1.83155948+1.60822041j
-1.83155948-1.60822041j 2.08155948+1.65109876j]
ifft:
[1. +0.j 2. +0.j 1. +0.j -1. +0.j 1.5+0.j]
可以看到fft,ifft返回的都是复数。ifft返回的结果中,复数的虚部都是0,实部与原始数据x⼀致。
这些点的频率⽆法计算,因为没有设置这N个点的时间长度。如不理解,不必深究,后⾯会介绍。
理解fft变换结果
我们知道,傅⾥叶变换把时域信号变为频域信号。在离散傅⾥叶变换中,频域信号由⼀系列不同频率的谐波(频率成倍数)组成。fft返回值是⼀个复数数组,每个复数表⽰⼀个正弦波。通常⼀个波形由振幅,相位,频率三个变量确定,可以从fft的返回值⾥,获取这些信息。
假设a是时域中的周期信号,采样频率为Fs,采样点数为N。如果A[N] = fft(a[N]),返回值A[N]是⼀个复数数组,其中:
A[0]表⽰频率为0hz的信号,即直流分量。
A[1:N/2]包含正频率项,A[N/2:]包含负频率项。正频率项就是转化后的频域信号,通常我们只需要正频率项,即前⾯的n/2项,负频率项是计算的中间结果(正频率项的镜像值)。
每⼀项的频率计算:假设A[i]为数组中的元素,表⽰⼀个波形,该波形的频率 = i * Fs / N
A[i] = real + j * imag,是⼀个复数,相位就是复数的辐⾓,相位 = arg(real/imag)
类似的,振幅就是复数的模,振幅 = sqrt(real^2+imag^2)。但是fft的返回值的模是放⼤值,直流分量的振幅放⼤了N倍,弦波分量的振幅放⼤了N/2倍。
频率分辨率
频率分辨率是离散傅⾥叶变换(DFT)频域相邻刻度之间的实际频率之差。采样时,数据采样了T秒(T = 采样点数N / 采样频率Fs),信号的成分中周期最⼤也就是T秒,最低频率即“基频”就等于1 / T,也就是Fs / N,这就是频率分辨率。基频 = Fs / N,各个谐波的频率就是 i * Fs / N,这个公式⽤于计算各个波形的频率。
⽰例
importnumpy as npfrom scipy.fftpack importfft#采样点数
N = 4000
#采样频率 (根据采样定理,采样频率必须⼤于信号最⾼频率的2倍,信号才不会失真)
Fs = 8000x= np.linspace(0.0, N/Fs, N)#时域信号,包含:直流分量振幅1.0,正弦波分量频率100hz/振幅2.0, 正弦波分量频率
150Hz/振幅0.5/相位np.pi
y = 1.0 + 2.0 * np.sin(100.0 * 2.0*np.pi*x) + 0.5*np.sin(150.0 * 2.0*np.pi*x +np.pi)#进⾏fft变换
yf =fft(y)#获取振幅,取复数的绝对值,即复数的模
abs_yf =np.abs(yf)#获取相位,取复数的⾓度
angle_y=np.angle(yf)#直流信号
print('\n直流信号')print('振幅:', abs_yf[0]/N) #直流分量的振幅放⼤了N倍
#100hz信号
index_100hz = 100 * N // Fs #波形的频率 = i * Fs / N,倒推计算索引:i = 波形频率 * N / Fs
print('\n100hz波形')print('振幅:', abs_yf[index_100hz] * 2.0/N) #弦波分量的振幅放⼤了N/2倍
print('相位:', angle_y[index_100hz])#150hz信号
index_150hz = 150 * N // Fs #波形的频率 = i * Fs / N,倒推计算索引:i = 波形频率 * N / Fs
print('\n150hz波形')print('振幅:', abs_yf[index_150hz] * 2.0/N) #弦波分量的振幅放⼤了N/2倍
print('相位:', angle_y[index_150hz])print('100hz与150hz相位差:', angle_y[index_150hz] -angle_y[index_100hz])print('\n')
输出
直流信号
振幅:1.0100hz波形
振幅:1.9989359813189005相位:-1.5315264186250062150hz波形
振幅:0.5008489983048182相位:1.6297011890497097100hz与150hz相位差:3.161227607674716
可以看到,正弦波的相位不⼀定从0开始,但波形之间的相位差确实s约等于⼀个pi(值跟采样频率与采样点数有关系)。
离散余弦变换(DCT)
由于许多要处理的信号都是实信号,在使⽤FFT时,对于实信号,傅⽴叶变换的共轭对称性导致在频域中有⼀半的数据冗余。
离散余弦变换(DCT)是对实信号定义的⼀种变换,变换后在频域中得到的也是⼀个实信号,相⽐离散傅⾥叶变换DFT⽽⾔, DCT可以减少⼀半以上的计算。DCT还有⼀个很重要的性质(能量集中特性):⼤多书⾃然信号(声⾳、图像)的能量都集中在离散余弦变换后的低频部分,因⽽DCT在(声⾳、图像)数据压缩中得到了⼴泛的使⽤。由于DCT是从DFT推导出来的另⼀种变换,因此许多DFT的属性在DCT中仍然是保留下来的。
SciPy.fftpack中,提供了离散余弦变换(DCT)与离散余弦逆变换(IDCT)的实现。
⽰例
importnumpy as npfrom scipy.fftpack importdct,idct
y= dct(np.array([4., 3., 5., 10., 5., 3.]))print(y)
输出
[ 60. -3.48476592 -13.85640646 11.3137085 6.-6.31319305]
离散余弦逆变换(idct),是离散余弦变换(DCT)的反变换。
⽰例
importnumpy as npfrom scipy.fftpack importdct,idct
y= idct(np.array([4., 3., 5., 10., 5., 3.]))print(y)
输出
[ 39.15085889 -20.14213562 -6.45392043 7.13341236 8.14213562 -3.83035081]

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。