web-dev-qa-db-fra.com

Calcul du spectrogramme des fichiers .wav en python

J'essaie de calculer le spectrogramme à partir de .wav fichiers utilisant Python. Dans un effort pour le faire, je suis les instructions qui peuvent être trouvées ici . Je suis d'abord lu .wav fichiers utilisant la bibliothèque librosa. Le code trouvé dans le lien fonctionne correctement. Ce code est:

sig, rate = librosa.load(file, sr = None)
sig = buf_to_int(sig, n_bytes=2)
spectrogram = sig2spec(rate, sig)

Et la fonction sig2spec:

def sig2spec(signal, sample_rate):

# Read the file.
# sample_rate, signal = scipy.io.wavfile.read(filename)
# signal = signal[0:int(1.5 * sample_rate)]  # Keep the first 3.5 seconds
# plt.plot(signal)
# plt.show()

# Pre-emphasis step: Amplification of the high frequencies (HF)
# (1) balance the frequency spectrum since HF usually have smaller magnitudes compared to LF
# (2) avoid numerical problems during the Fourier transform operation and
# (3) may also improve the Signal-to-Noise Ratio (SNR).
pre_emphasis = 0.97
emphasized_signal = numpy.append(signal[0], signal[1:] - pre_emphasis * signal[:-1])
# plt.plot(emphasized_signal)
# plt.show()
# Consequently, we split the signal into short time windows. We can safely make the assumption that
# an audio signal is stationary over a small short period of time. Those windows size are balanced from the
# parameter called frame_size, while the overlap between consecutive windows is controlled from the
# variable frame_stride.

frame_size = 0.025
frame_stride = 0.01
frame_length, frame_step = frame_size * sample_rate, frame_stride * sample_rate  # Convert from seconds to samples
signal_length = len(emphasized_signal)
frame_length = int(round(frame_length))
frame_step = int(round(frame_step))
num_frames = int(numpy.ceil(float(numpy.abs(signal_length - frame_length)) / frame_step))
# Make sure that we have at least 1 frame

pad_signal_length = num_frames * frame_step + frame_length
z = numpy.zeros((pad_signal_length - signal_length))
pad_signal = numpy.append(emphasized_signal, z)
# Pad Signal to make sure that all frames have equal
# number of samples without truncating any samples from the original signal

indices = numpy.tile(numpy.arange(0, frame_length), (num_frames, 1)) \
          + numpy.tile(numpy.arange(0, num_frames * frame_step, frame_step), (frame_length, 1)).T

frames = pad_signal[indices.astype(numpy.int32, copy=False)]

# Apply hamming windows. The rationale behind that is  the assumption made by the FFT that the data
# is infinite and to reduce spectral leakage.
frames *= numpy.hamming(frame_length)

# Fourier-Transform and Power Spectrum
nfft = 2048
mag_frames = numpy.absolute(numpy.fft.rfft(frames, nfft))  # Magnitude of the FFT
pow_frames = ((1.0 / nfft) * (mag_frames ** 2))  # Power Spectrum

# Transform the FFT to MEL scale
nfilt = 40
low_freq_mel = 0
high_freq_mel = (2595 * numpy.log10(1 + (sample_rate / 2) / 700))  # Convert Hz to Mel
mel_points = numpy.linspace(low_freq_mel, high_freq_mel, nfilt + 2)  # Equally spaced in Mel scale
hz_points = (700 * (10 ** (mel_points / 2595) - 1))  # Convert Mel to Hz
bin = numpy.floor((nfft + 1) * hz_points / sample_rate)

fbank = numpy.zeros((nfilt, int(numpy.floor(nfft / 2 + 1))))
for m in range(1, nfilt + 1):
    f_m_minus = int(bin[m - 1])  # left
    f_m = int(bin[m])  # center
    f_m_plus = int(bin[m + 1])  # right

    for k in range(f_m_minus, f_m):
        fbank[m - 1, k] = (k - bin[m - 1]) / (bin[m] - bin[m - 1])
    for k in range(f_m, f_m_plus):
        fbank[m - 1, k] = (bin[m + 1] - k) / (bin[m + 1] - bin[m])
filter_banks = numpy.dot(pow_frames, fbank.T)
filter_banks = numpy.where(filter_banks == 0, numpy.finfo(float).eps, filter_banks)  # Numerical Stability
filter_banks = 20 * numpy.log10(filter_banks)  # dB

return (filter_banks/ np.amax(filter_banks))*255

Je peux produire des images qui ressemblent à:

enter image description here

Cependant, dans certains cas, mon spectrogramme ressemble à:

Il se passe quelque chose de vraiment bizarre car au début du signal, il y a des bandes bleues dans les images que je ne comprends pas si elles signifient vraiment quelque chose ou s'il y a une erreur lors du calcul du spectrogramme. Je suppose que le problème est lié à la normalisation, mais je ne suis pas sûr de ce que c'est exactement.

enter image description here

EDIT: J'ai essayé d'utiliser la librosa recommandée de la bibliothèque:

sig, rate = librosa.load("audio.wav", sr = None)
spectrogram = librosa.feature.melspectrogram(y=sig, sr=rate)
spec_shape = spectrogram.shape
fig = plt.figure(figsize=(spec_shape), dpi=5)
lidis.specshow(spectrogram.T, cmap=cm.jet)
plt.tight_layout()
plt.savefig("spec.jpg")

La spécification est maintenant presque partout bleu foncé:

enter image description here

10
Jose Ramon

Cela peut être dû au fait que vous n'avez pas modifié les paramètres de la méthode du melspectrogramme librosa.

Dans votre implémentation d'origine, vous spécifiez nfft = 2048. Cela pourrait être transmis à melspectrogram et vous verrez des résultats différents.

Cet article décrit la "résolution de fréquence de la forme d'onde" et la "résolution fft" qui sont des paramètres importants lors de l'exécution d'un FT. Les comprendre peut aider à reproduire vos spectogrammes originaux.

http://www.bitweenie.com/listings/fft-zero-padding/

La méthode specshow a également divers paramètres qui affecteront directement les tracés que vous produisez.

cette pile de messages répertorie les différents paramètres du spectogramme dans MATLAB, mais vous pouvez dessiner des similitudes entre la version MATLAB et la version librosa.

Qu'est-ce qu'un spectrogramme et comment définir ses paramètres?

5
zoulzubazz