MATLAB 和 Engee 在频谱和时频分析任务中的比较
比较 MATLAB 和 Engee 在频谱分析和时频分析领域的功能,让我们来看看它们在正反傅立叶变换和时频分析这两项任务中的能力和方法。
让我们设置一个随机输入信号。
In [ ]:
Pkg.add(["DSP"])
In [ ]:
x_in = randn(300)
plot(x_in)
Out[0]:
傅立叶频谱变换(FFT 和 IFFT)
恩格**使用 FFTW 软件包中的函数:
- fft:直接快速傅立叶变换。
- ifft:反傅立叶变换。
In [ ]:
using FFTW
# Прямое преобразование Фурье
X = fft(x_in)
# Обратное преобразование Фурье
x_e = ifft(X)
Out[0]:
MATLAB 具有快速傅里叶变换的内置函数:
- fft:正向快速傅里叶变换。
- ifft:反快速傅立叶变换。
In [ ]:
using MATLAB
# Прямое преобразование Фурье
mat"""X = fft($(x_in));"""
# Обратное преобразование Фурье
x_m=mat"""ifft(X);"""
Out[0]:
让我们比较一下结果并计算误差。
In [ ]:
plot(x_in)
plot!(real(x_e))
plot!(x_m)
Out[0]:
In [ ]:
error_e = abs.(x_in-real(x_e))
error_m = abs.(x_in-x_m)
println("Средняя ошибка Engee: $(sum(error_e)/length(error_e))")
println("Средняя ошибка MATLAB: $(sum(error_m)/length(error_m))")
在这两种环境下,正向傅里叶变换的结果都是复数,而反向变换返回的是原始信号,由于数值舍入,误差很小。
由此可见,代码的语法和逻辑是完全相同的。
时频傅里叶变换
Engee 中没有此类变换的标准函数,必须手动实现。
In [ ]:
using DSP
# Функция для генерации чирп-сигнала
function chirp(t::AbstractVector{T}, f0::T, t1::T, k::T) where T<:Real
return cos.(2π * (f0 .* t .+ 0.5 * k .* t.^2))
end
# Окно Хэмминга
function hamming(N::Int)
n = 0:N-1
return 0.54 .- 0.46 .* cos.(2π * n / (N - 1))
end
x = chirp(0:0.001:1, 0.0, 1.0, 100.0) # Чирп-сигнал
l=500 # Размер окна
window=vcat(hamming(l),zeros(length(x)-l))
P = DSP.periodogram(x; fs=1000, window=window, nfft=length(x))
# Визуализация спектра
plot(freq(P), 20*log10.(power(P)), xlabel="Frequency (Hz)", ylabel="Power Spectrum (dB)", title="Power Spectrum", linewidth=2)
Out[0]:
MATLAB 有一个 pspectrum 函数,可以对信号进行时频分析。
pspectrum 会自动选择适当的分析方法。
它支持各种可视化(频谱图、功率密度)。
In [ ]:
using Images
mat"""cd("$(@__DIR__)")"""
mat"""
% Частотно-временной анализ
x = chirp(0:0.001:1, 0, 1, 100);
pspectrum(x, 1000); % Частотно-временной спектр
saveas(gcf, 'frequency_time_spectrum.jpg'); % Сохраняет текущую фигуру как JPG
"""
load( "$(@__DIR__)/frequency_time_spectrum.jpg" )
Out[0]:
从图中可以看出,Engee 反映了信号行为的主要趋势。要进行更详细的绘图,还需要对周期图函数进行更详细的窗口计算。
我们还可以比较一下 MATLAB 和 Engee 中生成的输入数据的正确性。
In [ ]:
plot(mat"chirp(0:0.001:1, 0, 1, 100)"[:])
plot!(chirp(0:0.001:1, 0.0, 1.0, 100.0))
title!("Sum_error: $(sum(mat"chirp(0:0.001:1, 0, 1, 100)"[:]-chirp(0:0.001:1, 0.0, 1.0, 100.0)))")
Out[0]:
我们可以看到,输入信号完全相同。
输出
MATLAB 内置了执行这些操作的函数。在 Engee 中,您可以使用第三方库(如 DSP 或 FFTW)进行更复杂的信号操作。同样重要的是,Engee 在如何实现转换方面提供了更大的灵活性。为了简化 Engee 实现中的这些操作,请使用本演示中介绍的开箱即用方法。
比较两种环境功能的另一个方面是,MATLAB 针对矩阵和信号进行了优化。不过,如果在编写代码时考虑到这一环境的特殊性,Engee 的计算速度通常会更快,这将显著提高大型系统中算法的执行和测试速度。