目标
我希望使用具有复 Morlet(又名 Gabor)小波的实值信号的连续小波变换(CWT)来计算相干估计。我以与pycwt 库中非常相似的方式计算信号的 cwt ,但我使用 pyfftw 库进行更快的 FFT 计算。我相信 cwt 的计算是正确的,因为我得到了几乎相同pycwt.cwt
的结果,并且 scalograms 与 STFT 频谱图相比很好。虽然我使用 Python,但我认为它不是特定于语言的。
相干性计算资源
但是,我对如何计算相干性估计没有那么自信。我的相干性计算基于经典的Torrence & Webster 1999论文(与 pycwt 一样),其中在计算相干性分数之前,他们使用标准偏差等于的高斯核及时平滑交叉功率密度 ( cwt(sig)1 * cwt(sig2).conj()
) 和功率密度 ( )abs(cwt(sig))**2
到每个尺度的尺度。这对我来说很有意义,因为对于较大的尺度,光谱特性的变化速度较慢,因此与较小的尺度相比,更大的平滑窗口是必要的。他们还使用简单的移动平均线在比例域中平滑它们,这对我来说意义不大。
定义和实现的问题
这是我寻求解释和/或帮助的平滑运算符的实现和/或定义的问题
时域平滑
尺度域中的平滑
- 尺度域中的移动平均平滑似乎是经验性的,但我可以接受。我只是现在似乎有真正的解释为什么它是必要的(据我所知,韦尔奇的相干估计方法没有这样做)。是否有充分的理由在比例域中执行平滑?
- 真正让我困惑的是pycwt在对数间隔的尺度上计算它,这是一个非线性滤波器,我不确定这是否有意义。即使作为非线性滤波器也有意义吗?
到目前为止的结果
到目前为止,我只在时域中使用了一个简单的移动平均滤波器,并且在低频以上给出了合理的结果。但是,我对结果没有信心,并且将来我需要较低频率的相干性,所以我在这里问。我还尝试使用具有更大标准偏差的高斯窗口来获得类似于移动平均窗口平滑结果的结果。
每个第一个子图中的粗线是原始半透明信号的低通滤波版本。我添加了这些低通信号,以便更容易区分这两个信号。
使用对数频率轴(更好地显示低频):
使用线性频率轴(显示一般一致性):
这是我用于平滑的代码部分,fftmod
是pyfftw.interfaces.numpy_fft
(提供与模块相同的功能numpy.fft
)
# smoothing in time domain
freqs = fftmod.fftfreq(self.time_axis.data.shape[0],
self.axis_step(0))[:,np.newaxis]
freqs0 = self.axis1.data
if smoothing_window: # moving average
if isinstance(smoothing_window, complex):
# in time units, convert to number of points
smoothing_window = abs(smoothing_window) / self.axis_step(0)
smoothing_window = np.empty(smoothing_window)[:,np.newaxis]
smoothing_window.fill(1.0 / smoothing_window.size)
smoothing_ft = fftmod.fft(smoothing_window, n=freqs.size,
axis=0, **fftw_kwargs)
else: # gaussian smoothing, gaussian_std_mul multiplies the std dev
# FT of smoothing window which is a gaussian with std dev 1/f0
# is a gaussian with std dev f0 without f0 in normalization const
smoothing_ft = np.exp(-(freqs/freqs0 * gaussian_std_mul)**2 / 2.0) / _SQRT_2PI
# smooth by convolution with window as multiplication in Fourier domain
csd = fftmod.ifft(smoothing_ft * fftmod.fft(
csd, axis=0, **fftw_kwargs), axis=0, **fftw_kwargs) # TODO maybe fft of vstack
psd_self = fftmod.ifft(smoothing_ft * fftmod.fft(
psd_self, axis=0, **fftw_kwargs), axis=0, **fftw_kwargs).real
psd_signal = fftmod.ifft(smoothing_ft * fftmod.fft(
psd_signal, axis=0, **fftw_kwargs), axis=0, **fftw_kwargs).real
更多最新资源
我做了一些进一步的挖掘,发现了 Bernard Cazelles 2007 年的一篇文章。他网站上的代码(包括 zip 文件中的文章)似乎表明他还在尺度和频域中使用了各种不同窗口(Barttlet、Hamming、Boxcar...)的平滑,但确实使用了不同的时域中不同尺度的窗口长度,尺度域中只有一个长度。该代码显示比例域再次是对数的,但使用线性窗口进行了平滑处理。然而,至少他引用了 Chatfield 1989,他引用了 PJ Daniell 在 1946 年的建议,即频域中的平滑也是需要的。