Source code for mosqito.sound_level_meter.noct_spectrum.noct_spectrum
# -*- coding: utf-8 -*-
# Standard library imports
from numpy import newaxis, array, squeeze
# Local application imports
from mosqito.sound_level_meter.noct_spectrum._filter_bandwidth import _filter_bandwidth
from mosqito.sound_level_meter.noct_spectrum._n_oct_time_filter import (
_n_oct_time_filter,
)
from mosqito.sound_level_meter.noct_spectrum._center_freq import _center_freq
[docs]
def noct_spectrum(sig, fs, fmin, fmax, n=3, G=10, fr=1000):
"""Compute nth-octave band spectrum
This function computes the rms level of a signal for each third octave band
between the 2 limit frequencies.
Parameters
----------
sig : array_like
Time signal array with size (nperseg, nseg).
fs : float
Sampling frequency [Hz]
fmin : float
Minimum frequency band [Hz]
fmax : float
Maximum frequency band [Hz]
n : int
Number of bands per octave
G : int
System for specifying the exact geometric mean frequencies.
Can be base 2 or base 10
fr : int
Reference frequency. Shall be set to 1 kHz for audible frequency
range, to 1 Hz for infrasonic range (f < 20 Hz) and to 1 MHz for
ultrasonic range (f > 31.5 kHz)
Returns
-------
spec : array_like
nth-octave octave band spectrum of signal sig with size (nfreq, nseg)
fpref : array_like
Corresponding prefered nth-octave octave band center frequencies
See Also
--------
.comp_spectrum : Spectrum computation from a time signal
.noct_synthesis : Conversion of a spectrum to n-th octave band levels
Examples
--------
.. plot::
:include-source:
>>> from mosqito.sound_level_meter import noct_spectrum
>>> from mosqito.utils import amp2db
>>> import matplotlib.pyplot as plt
>>> import numpy as np
>>> f=1000
>>> fs=48000
>>> d=0.2
>>> dB=60
>>> time = np.arange(0, d, 1/fs)
>>> stimulus = np.sin(2 * np.pi * f * time) + 0.5 * np.sin(6 * np.pi * f * time)
>>> rms = np.sqrt(np.mean(np.power(stimulus, 2)))
>>> ampl = 0.00002 * np.power(10, dB / 20) / rms
>>> stimulus = stimulus * ampl
>>> spec, freq_axis = noct_spectrum(stimulus, fs, fmin=90, fmax=14000)
>>> spec_db = amp2db(spec, ref=2e-5)
>>> plt.step(freq_axis, spec_db)
>>> plt.xlabel("Center frequency [Hz]")
>>> plt.ylabel("Amplitude [dB]")
"""
# 1-dimensional array to 2-dimensional array with size (nperseg, 1)
if sig.ndim == 1:
sig = sig[:, newaxis]
# Get filters center frequencies
fc_vec, fpref = _center_freq(fmin=fmin, fmax=fmax, n=n, G=G, fr=fr)
# Compute the filters bandwidth
alpha_vec, _, _ = _filter_bandwidth(fc_vec, n=n)
# Calculation of the rms level of the signal in each band
spec = []
for fc, alpha in zip(fc_vec, alpha_vec):
spec.append(_n_oct_time_filter(sig, fs, fc, alpha))
return squeeze(array(spec)), fpref