8
7

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

Pythonその4Advent Calendar 2020

Day 18

パワースペクトルから振動振幅を求めるために

Last updated at Posted at 2019-12-07

波形解析(揺動解析・振動解析)で高速フーリエ変換する場合、
しばしば規格化する必要があるので、係数に無関心になりがち。
振動振幅を求めるときに係数が分からないといけないのでまとめておきます。

####サンプル波形の作成 (準備)

python3.6.py
import numpy as np
import matplotlib.pyplot as plt

#サンプル波形の作成
def sample_wave(dlen,fs,dt,a0,f0,x0):
    #  dat = x0 + a sin( 2pi f t ) + noise(0,1)
    t    = np.arange(dlen) * dt              #時間データ [sec]
    dat  = a0 * np.sin(2.* np.pi * f0 * t)   #sin波形
    dat += x0 + np.random.normal(0.,1.,dlen) #中心値0, 分散1のホワイトノイズ
    return t,dat

#パラメータ入力
dlen  = 2**16  # データのデータ長   [points]
fs    = 500.e3 # サンプリング周波数 [Hz]
dt    = 1./fs  # サンプリング間隔   [sec]
a0    = 1.     #振幅       [V]
f0    = 1000.  #周波数     [Hz]
x0    = 10.0   #オフセット [V]

t,dat = sample_wave(dlen,fs,dt,a0,f0,x0)
plt.plot(t,dat,".")
plt.xlim([0,0.01])

image.png

#1. FFT (scipy.fftpack.fftの利用)
###単純に生波形をFFTにかける

python3.6.py
from scipy import fftpack
sxx  = fftpack.fft(dat)           #FFTの実行 (窓関数なし)
freq = fftpack.fftfreq(dlen,d=dt) #周波数リストの作成
df   = 1./(dlen*dt)               #周波数分解能        
Pxx  = np.abs(sxx)**2 / fs /dlen  #パワースペクトラム [V**2/Hz]

plt.plot(freq,Pxx,".") 

plt.xlabel("frequency [Hz]")
plt.ylabel("power spectrum [V**2/Hz]")
plt.xscale("log")
plt.yscale("log")                      #負の周波数は非表示になる

image.png

1000Hzのパワーピーク確認。その値を調べて、振幅を見積もる。

python3.6.py
def getNearestindex(list, num):
    """
    概要: リストからある値に最も近いインデックスを返す関数
    @param list: データ配列
    @param num: 対象値
    @return 対象値に最も近い値
    """
    # リスト要素と対象値の差分を計算し最小値のインデックスを取得
    idx = np.abs(np.asarray(list) - num).argmin()
    return idx

ind = getNearestindex(freq,f0)     #f0に近いインデックスを取得
print("ind=",ind,"val=",freq[ind]) 
#>>    ind=  131  val=  999.4

#f0の振幅について
# パワー密度[V**2/Hz]に[Hz]をかけてRMSを求める
rms = np.sqrt( Pxx[ind] * df * 2 ) # 負の周波数のパワーも考慮して2倍
amp = rms * np.sqrt(2)             # 振幅= 実効値* sqrt(2)
print("amplitude =", amp )
#>>    amplitude = 0.9920      # a0 = 1.0と少し離れている

#f0周辺の周波数バンドの振幅について(ind+/-2の領域を取った)
#バンド内のパワー密度の総和でバンド内のRMSを求める
rms = np.sqrt( np.sum(Pxx[ind-2:ind+3]) *df * 2 )  # 負の周波数のパワーも考慮して2倍
amp = rms * np.sqrt(2)                             # 振幅= 実効値* sqrt(2)                          
print("amplitude =", amp )
#>>    amplitude = 0.9982         << a0 = 1.0に近づいた

###生波形を規格化してFFTにかける
次はオフセットを引き、オフセットの値で規格化した方法

python3.6.py
t,datn = sample_wave(dlen,fs,dt,a0,f0,x0) #パラメータは前項と同じ
datn   = (dat-x0) / x0                    #規格化

plt.plot(t,datn,".")
plt.xlim([0,0.01])
plt.xlabel("time [sec]")
plt.ylabel("data [V]")

image.png

python3.6.py
sxxn  = fftpack.fft(datn)
freq  = fftpack.fftfreq(dlen,d=dt)
df    = 1./(dlen*dt)
Pxx   = np.abs(sxxn)**2 / fs /dlen  #パワースペクトラム

次元が [V] => [(V-x0)/x0] に変わったことに注意して、
1000Hzのパワーピーク確認。その値を調べて、振幅を見積もる。

python3.6.py
#f0周辺の周波数バンドの振幅について(ind+/-2の領域を取った)
#バンド内のパワー密度の総和でバンド内のRMSを求める
rms = np.sqrt( np.sum(Pxx[ind-2:ind+3]) *df *2 ) #負の周波数のパワーも考慮して2倍
amp = rms * np.sqrt(2)                                 
print("amplitude =", amp )
#>>    amplitude = 0.0998         << a0/x0 = 1.0/10.0 = 0.1に近い

#2.FFT (scipy.signalの利用)
scipy.signalには高級ルーチンがあるので素性を確かめる。
分かりやすさのため、あえて全体のデータ長とwindow長は同じにする

signal.stft と  singal.spectrogram の比較

python3.6.py
from scipy import signal
t,dat = sample_wave(dlen,fs,dt,a0,f0,x0) #パラメータは前項と同じ

Sxx,frq,Pxx={},{},{}

#fftpack.fft #前項と同じ
frq["fft"],  _t, Sxx["fft"]  = fftpack.fftfreq(dlen,d=dt), _, fftpack.fft( dat )

#signal.stft        (窓関数を矩形, window = "boxcar")
frq["stft"], _t, Sxx["stft"] = \
    signal.stft(      dat,fs       = fs,
                          nperseg  = dlen,
                          window   = "boxcar",
                          boundary = None,
                          detrend  = "constant",  
                          return_onesided = False   )

#singal.spectrogram (窓関数を矩形, mode="complex", scaling='spectrum')
frq["spectrogram_spectrum"],_t,Sxx["spectrogram_spectrum"]  = \
    signal.spectrogram(dat,fs      = fs,
                           nperseg = dlen,
                           mode    = "complex",
                           window  = "boxcar",
                           scaling = "spectrum",
                           return_onesided = False   )

#singal.spectrogram (窓関数を矩形,  mode="psd", scaling='density')
frq["spectrogram_density"],_t,Sxx["spectrogram_density"]  = \
    signal.spectrogram(dat,fs      = fs,
                           nperseg = dlen,
                           mode    = "psd",
                           window  = "boxcar",
                           scaling = "density",
                           return_onesided = False   )

####スペクトラム密度 [ V**2/Hz ] に係数に気を付けて変換
Pxx["fft"]                   = np.abs( Sxx["fft"]                       )**2 / fs / dlen
Pxx["stft"]                  = np.abs( Sxx["stft"][:,0]                 )**2 / df        
Pxx["spectrogram_spectrum"]  = np.abs( Sxx["spectrogram_spectrum"][:,0] )**2 / df
Pxx["spectrogram_density" ]  = np.abs( Sxx["spectrogram_density" ][:,0] )

#前項のscipy.fftpack.fftと似てるかどうか
for key in ["fft","stft","spectrogram_spectrum","spectrogram_density"]: 
    print(np.allclose( Pxx["fft"][1:], Pxx[key][1:], atol=1e-4), key )
#>>                    True fft
#>>                    True stft
#>>                    True spectrogram_spectrum
#>>                    True spectrogram_density

freq=0以外一致します。

#### 窓関数の採用について

python3.6.py
window = ("tukey",0.25)
frq["spectrogram_spectrum_tukey"] ,t,Sxx["spectrogram_spectrum_tukey" ] = \
    signal.spectrogram(dat,fs      = fs,
                           nperseg = dlen,
                           return_onesided = False,
                           mode    = "complex",
                           window  = window,
                           scaling = 'spectrum'         )

window = ("tukey",0.25)
frq["spectrogram_density_tukey"] ,t,Sxx["spectrogram_density_tukey" ] = \
    signal.spectrogram(dat,fs      = fs,
                           nperseg = dlen,
                           return_onesided = False,
                           mode    = "psd",
                           window  = window,
                           scaling = 'density'         )

Pxx["spectrogram_spectrum_tukey" ]  = np.abs( Sxx["spectrogram_spectrum_tukey"][:,0] )**2 / df
Pxx["spectrogram_density_tukey" ]   = np.abs( Sxx["spectrogram_density_tukey" ][:,0] )

#グラフで比較する
for key in ["fft","spectrogram_spectrum_tukey","spectrogram_density_tukey"]:
    plt.plot(frq[key],np.abs(Pxx[key]),".",label=key)
plt.xlabel("frequency [Hz]")
plt.ylabel("power spectrum [V**2/Hz]")
plt.yscale("log")
plt.xscale("log")
plt.xlim([100,5000])
plt.ylim([1e-9,1e0])
plt.legend()

image.png
窓関数のサイドローブがあるので厳密には一致しないが、窓関数補正もかかっている。

#### return_oneside =True/Falseについて

python3.6.py
window = ("tukey",0.25)
key = "spectrogram_psd_tukey_oneside"
frq[key] ,t,Sxx[key] = \
    signal.spectrogram(dat,fs      = fs,
                           nperseg = dlen,
                           return_onesided = True,
                           mode    = "psd",
                           window  = window,
                           scaling = 'density'         )
Pxx[key] = np.abs( Sxx[key][:,0] )

#振幅で確認
key = "spectrogram_density"
print( np.sqrt( np.sum(Pxx[key][ind-3:ind+4]) * 2 * df)*np.sqrt(2), key) #負の周波数を考慮して2倍
#>> 0.987 spectrogram_density
key = "spectrogram_psd_tukey_oneside"
print( np.sqrt( np.sum(Pxx[key][ind-3:ind+4]) *     df)*np.sqrt(2), key) #すでに折込み済み
#>> 0.987 spectrogram_psd_tukey_oneside

一致しています。

#まとめ

function mode scaling onside how to cal. Power
fftpack.fft() (complex) (spectrum) (False) S^2/ fs/ dlen * 2
stft() (complex) (spectrum) False S^2/ df * 2
stft() (complex) (spectrum) True S^2/ df * 2
spectrogram() complex spectrum False S^2/ df * 2
spectrogram() psd density False S *2
spectrogram() psd density True S
*Sは関数の戻り値
*窓関数は補正してくれる

rms = sqrt(Power * df )
振幅 = rms * sqrt(2)

説明 本稿の記号
データ長 dlen
サンプリングレート fs
時間分解能 dt = 1./fs
周波数分解能 df = 1./(dlen*dt) = 1./ T

####よくみたソースコード
scipy.fftpack.fft
scipy.signal.stft
scipy.signal.spectrogram
_spectral_helperと_fft_helper

####あとがき
stft()とfftpack.fft()の出力が違うことでハマった。
毎回係数で不安になるので備忘録代わりに一度まとめておきました。

8
7
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
8
7

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?