功率谱密度的相关推导以及Python实现

本文主要介绍了离散信号功率谱密度的相关推导以及 P y t h o n Python Python实现。特别是,很多教材默认采样频率为单位1,本文不做此默认相关推导更具一般性。文章内容安排如下:第一部分介绍基本概念和相关推导;第二部分分别利用现成的 m a t p l o t l i b . p y p l o t . p s d matplotlib.pyplot.psd matplotlib.pyplot.psd库和 n u m p y . f f t . f f t numpy.fft.fft numpy.fft.fft库计算离散信号的功率谱密度并验证结果。湍流领域中的文献常用预乘谱,其物理解释可以参考这个网站。功率谱密度的现代估计方法可以参考这个网站。

一、理论推导

1.1 基本概念

对于一个连续随机信号 x c ( t ) x_c(t) xc(t),由帕萨维尔等式可知:
∫ − ∞ + ∞ x c ( t ) 2 d t = 1 2 π ∫ − ∞ + ∞ ∣ x ^ c ( ω ) ∣ 2 d ω (1) \int_{-\infty}^{+\infty}x_c(t)^2dt = \frac{1}{2\pi}\int_{-\infty}^{+\infty}|\hat{x}_c(\omega)|^2d\omega \tag{1} +xc(t)2dt=2π1+x^c(ω)2dω(1) 其中, x ^ c ( ω ) \hat{x}_c(\omega) x^c(ω)为随机信号 x c ( t ) x_c(t) xc(t)的傅里叶变换。类似于电磁学中用电流信号的平方表示功率( P = I 2 R P=I^2R P=I2R),上式左端被积函数 W = x c ( t ) 2 W=x_c(t)^2 W=xc(t)2表示随机信号的功率。因此,(1)式左端表示随机信号的总能量,右端被积函数 E ( ω ) = ∣ x ^ c ( ω ) ∣ 2 E(\omega)=|\hat{x}_c(\omega)|^2 E(ω)=x^c(ω)2表示单位频率的能量,也即连续信号的能量谱密度。对于离散随机信号 x ( n ) = x c ( n Δ t ) , x ^ ( k ) = x ^ c ( ω k ) x(n)=x_c(n\Delta{t}),\hat{x}(k)=\hat{x}_c(\omega_k) x(n)=xc(nΔt),x^(k)=x^c(ωk),有离散形式的帕萨维尔等式:
∑ n = 0 N − 1 x c ( n Δ t ) 2 Δ t = 1 2 π ∑ k = 0 N − 1 ∣ x ^ c ( ω k ) ∣ 2 Δ ω (2) \sum_{n=0}^{N-1}x_c(n\Delta{t})^2 \Delta{t}= \frac{1}{2\pi}\sum_{k=0}^{N-1}|\hat{x}_c(\omega_k)|^2\Delta{\omega}\tag{2} n=0N1xc(nΔt)2Δt=2π1k=0N1x^c(ωk)2Δω(2) ω k = 2 π f k = 2 k π T = 2 k π N Δ t , Δ ω = 2 π Δ f = 2 π N Δ t . \omega_k=2\pi{f_k}=\frac{2k\pi}{T}=\frac{2k\pi}{N\Delta{t}},\Delta{\omega=2\pi\Delta{f}=\frac{2\pi}{N\Delta{t}}}. ωk=2πfk=T2=NΔt2,Δω=2πΔf=NΔt2π.其中, Δ t \Delta{t} Δt为采样周期,相应的采样频率为: f s = 1 / Δ t . f_s=1/\Delta{t}. fs=1/Δt. 将上式整理得到:
∑ n = 0 N − 1 x ( n ) 2 Δ t = ∑ k = 0 N − 1 ∣ f s ⋅ x ^ ( k ) ∣ 2 f s 2 ⋅ Δ f (3) \sum_{n=0}^{N-1}x(n)^2 \Delta{t}=\sum_{k=0}^{N-1}\frac{|f_s·\hat{x}(k)|^2}{f_s^2}·\Delta{f}\tag{3} n=0N1x(n)2Δt=k=0N1fs2fsx^(k)2Δf(3)式中 x ^ s ( k ) = f s ⋅ x ^ ( k ) \hat{x}_s(k)=f_s·\hat{x}(k) x^s(k)=fsx^(k)离散傅里叶变换的结果(推导见下节内容)。于是上式可以写成:
∑ n = 0 N − 1 x ( n ) 2 Δ t = ∑ k = 0 N − 1 ∣ x ^ s ( k ) ∣ 2 f s 2 ⋅ Δ f (4) \sum_{n=0}^{N-1}x(n)^2 \Delta{t}=\sum_{k=0}^{N-1}\frac{|\hat{x}_s(k)|^2}{f_s^2}·\Delta{f}\tag{4} n=0N1x(n)2Δt=k=0N1fs2x^s(k)2Δf(4)对于离散信号,我们可以定义离散信号的能量谱密度为离散信号离散傅里叶变换的模的平方除以采样频率的平方,即: E ( k ) = ∣ x ^ s ( k ) ∣ 2 / f s 2 . E(k)=|\hat{x}_s(k)|^2/f_s^2. E(k)=x^s(k)2/fs2. 此时,离散信号的平均功率为:
W ‾ = 1 N ∑ n = 0 N − 1 x ( n ) 2 = ∑ k = 0 N − 1 ∣ x ^ s ( k ) ∣ 2 N f s ⋅ Δ f . (5) \overline{W}=\frac{1}{N}\sum_{n=0}^{N-1}x(n)^2=\sum_{k=0}^{N-1}\frac{|\hat{x}_s(k)|^2}{Nf_s}·\Delta{f}.\tag{5} W=N1n=0N1x(n)2=k=0N1Nfsx^s(k)2Δf.(5)同理,可以定义离散信号的功率谱密度为离散信号的能量谱密度除以总时间T,即: P ( k ) = E ( k ) N Δ t = ∣ x ^ s ( k ) ∣ 2 / ( f s N ) P(k)=\frac{E(k)}{N\Delta{t}}=|\hat{x}_s(k)|^2/(f_sN) P(k)=NΔtE(k)=x^s(k)2/(fsN)。一般来说,我们需要对原始信号加窗 w ( n ) w(n) w(n)。加窗后,为使其平均功率保持不变需要乘上一个能量恢复系数 K K K[1],也即有: 1 N ∑ n = 0 N − 1 x ( n ) 2 = K ∗ 1 N ∑ n = 0 N − 1 w ( n ) 2 x ( n ) 2 . \frac{1}{N}\sum_{n=0}^{N-1}x(n)^2=K*\frac{1}{N}\sum_{n=0}^{N-1}w(n)^2x(n)^2. N1n=0N1x(n)2=KN1n=0N1w(n)2x(n)2.为计算方便,我们令 x ( n ) = 1 x(n)=1 x(n)=1得到加函数 w ( n ) w(n) w(n)窗的恢复系数为: K = N ∑ n = 0 N − 1 w ( n ) 2 = N ∣ ∣ w ∣ ∣ 2 . K=\frac{N}{\sum_{n=0}^{N-1} w(n)^2}=\frac{N}{||w||^2}. K=n=0N1w(n)2N=∣∣w2N.加窗以后的功率谱密度为: P ( k ) = K ∗ E ( k ) N Δ t = ∣ x ^ s ( k ) ∣ 2 f s ∣ ∣ w ∣ ∣ 2 . P(k)=K*\frac{E(k)}{N\Delta{t}}=\frac{|\hat{x}_s(k)|^2}{f_s||w||^2}. P(k)=KNΔtE(k)=fs∣∣w2x^s(k)2.

1.2 离散傅里叶变换

对连续信号 x c ( t ) x_c(t) xc(t)进行采样得到离散信号 x ( t ) x(t) x(t)的过程可以描述为:
x ( t ) = x c ( t ) s ( t ) , s ( t ) = ∑ n = 0 N − 1 δ ( t − n Δ t ) , δ ( t − t 0 ) = { 1 , t = t 0 0 , t ≠ t 0 (6) x(t)=x_c(t)s(t),s(t)=\sum_{n=0}^{N-1}\delta({t-n\Delta{t}}), \delta(t-t_0) = \left\{\begin{aligned} 1, & &t=t_0 \\ 0, & &t\neq{t_0}\\ \end{aligned}\right.\tag{6} x(t)=xc(t)s(t),s(t)=n=0N1δ(tnΔt),δ(tt0)={1,0,t=t0t=t0(6)其中, s ( t ) s(t) s(t)称之为采样函数。根据连续函数的傅里叶变换可知:
x ^ c ( ω ) = ∫ − ∞ + ∞ x c ( t ) e − i ω t d t . (7) \hat{x}_c(\omega)=\int_{-\infty}^{+\infty}x_c(t)e^{-i\omega{t}}dt.\tag{7} x^c(ω)=+xc(t)etdt.(7)将随机信号 x ( t ) x(t) x(t)代入上式即可得到离散时间傅里叶变换如下:
x ^ ( ω ) = ∫ − ∞ + ∞ x c ( t ) s ( t ) e − i ω t d t = ∑ n = 0 N − 1 ∫ − ∞ + ∞ x c ( t ) δ ( t − n Δ t ) e − i ω t d t . (8) \hat{x}(\omega)=\int_{-\infty}^{+\infty}x_c(t)s(t)e^{-i\omega{t}}dt=\sum_{n=0}^{N-1}\int_{-\infty}^{+\infty}x_c(t)\delta(t-n\Delta{t})e^{-i\omega{t}}dt.\tag{8} x^(ω)=+xc(t)s(t)etdt=n=0N1+xc(t)δ(tnΔt)etdt.(8)将上述积分写成离散形式可得:
x ^ ( ω ) = ∑ n = 1 N − 1 ∑ k = 0 N − 1 x c ( t ) δ ( k Δ t − n Δ t ) e − i ω k Δ t Δ t = ∑ n = 1 N − 1 x c ( n Δ t ) e − i ω n Δ t Δ t . (9) \hat{x}(\omega)=\sum_{n=1}^{N-1}\sum_{k=0}^{N-1}x_c(t)\delta(k\Delta{t}-n\Delta{t})e^{-i\omega{k\Delta{t}}}\Delta{t}=\sum_{n=1}^{N-1}x_c(n\Delta{t})e^{-i\omega{n\Delta{t}}}\Delta{t}.\tag{9} x^(ω)=n=1N1k=0N1xc(t)δ(kΔtnΔt)ekΔtΔt=n=1N1xc(nΔt)enΔtΔt.(9)将频率离散后(见上一节的处理)代入上述方程即可得到离散傅里叶变换如下:
x ^ ( ω k ) = ∑ n = 0 N − 1 x c ( n Δ t ) e − i ω k n Δ t Δ t . (10) \hat{x}(\omega_k)=\sum_{n=0}^{N-1}x_c(n\Delta{t})e^{-i\omega_k{n\Delta{t}}}\Delta{t}.\tag{10} x^(ωk)=n=0N1xc(nΔt)eiωknΔtΔt.(10)整理得到:
x ^ s ( k ) = f s ⋅ x ^ ( k ) = ∑ n = 0 N − 1 x ( n ) e − i 2 π k n N . (11) \hat{x}_s(k)=f_s·\hat{x}(k)=\sum_{n=0}^{N-1}x(n)e^{-i\frac{2\pi{kn}}{N}}.\tag{11} x^s(k)=fsx^(k)=n=0N1x(n)eiN2πkn.(11)等式左端 x ^ s ( k ) \hat{x}_s(k) x^s(k)即为离散傅里叶变换的结果。

二、python代码

  1. 导入基本库
import numpy as np
from numpy.fft import fft
from numpy.linalg import norm
import matplotlib.pyplot as plt
  1. 离散信号的生成
pi = np.pi  # 圆周率
dt = 0.1    # 采样周期
fs = 1/dt   # 采样频率

f1 = 0.05   # 信号的特征频率1
f2 = 0.40   # 信号的特征频率2
f3 = 1.0    # 信号的特征频率3

N  = 2**12                                                          # 离散信号的长度
tn = np.arange(0,dt*N,dt)                                           # 时间序列
x  = 2*np.cos(2*pi*f1*tn)+2*np.cos(2*pi*f2*tn)+2*np.cos(2*pi*f3*tn) # 生成离散信号
x  = x*(1+0.1*np.random.randn(N))                                   # 加入随机噪声  
nfft = 256                                                          # psd的窗长

# 将离散信号输出
plt.figure(figsize=(22,2))
plt.plot(tn,x,'c-',linewidth=0.5)
plt.xlim(0,dt*N)
plt.show()

输出结果如下图所示:
离散随机信号

  1. 利用plt.psd计算功率谱密度
[Pxx1,f1] = plt.psd(x,                       # 随机信号
                    NFFT=nfft,               # 每个窗的长度
                    Fs=fs,                   # 采样频率
                    detrend='mean',          # 去掉均值
                    window=np.hanning(nfft), # 加汉尼窗
                    noverlap=int(nfft*3/4),  # 每个窗重叠75%的数据
                    sides='twosided')        # 求双边谱
plt.xscale('log')

输出结果如下图所示:
plt.psd的结果

  1. 利用numpy库中的fft计算功率谱密度
Pxx = []
for i in range(N//nfft):                                      # 窗与窗之间数据不重叠
    w = np.hanning(nfft)                                      # 加汉尼窗
    p = np.abs(fft(w*x[i*nfft:(i+1)*nfft]))**2/fs/norm(w)**2  # 计算功率谱
    Pxx.append(p)                                             # 每个窗的结果
Pxx2 = np.mean(Pxx,axis=0)                                    # 将所有窗平均得到最终结果

f2   = np.array([k*fs/nfft for k in range(nfft//2)])          # 相应的频率:fk

将结果可视化并与plt.psd的结果进行比较。

# 双边谱只有半边有效(采样定理)
plt.plot(f2,Pxx2[:nfft//2],color='orange',linestyle='-',label='fft') # 利用fft计算的结果
plt.plot(f1[nfft//2:],Pxx1[nfft//2:],'b--',label='plt.psd')          # 利用plt.psd计算的结果

plt.xscale('log')
plt.yscale('log')
plt.xlabel('f/Hz',fontsize=12)
plt.ylabel('PSD', fontsize=12)
plt.legend(frameon=False,fontsize=12)
plt.tick_params(top=True,right=True,direction='in',which='both')
plt.show()

psd对比结果

  1. 结果验证
    按照前面的定义可知,功率谱密度与频率围成的面积表示离散随机信号的平均功率。
print('平均功率:',sum(x**2)/N)           # 输出离散随机信号的的平均功率
print(' plt.psd:',sum(Pxx1)*fs/nfft)    # 输出plt.psd计算得到的平均功率
print('     fft:',sum(Pxx2)*fs/nfft)    # 输出fft计算得到的平均功率

输出结果为:
验证结果
若有错误欢迎在下方评论指出,谢谢!
[1]焦新涛,丁康.加窗频谱分析的恢复系数及其求法[J].汕头大学学报(自然科学版),2003(03):26-30+38.

Logo

华为开发者空间,是为全球开发者打造的专属开发空间,汇聚了华为优质开发资源及工具,致力于让每一位开发者拥有一台云主机,基于华为根生态开发、创新。

更多推荐