Discrete and fast Fourier transforms
Modern measurement systems primarily construct spectra by sampling an analog electronic signal \(y(t)\) to yield the sample sequence \((y_n)\) and perform a discrete Fourier transform.
Definition
The discrete Fourier transform (DFT) of a sample sequence \((y_n)\) of length \(N\) is \((Y_m)\), where \(m \in [0,1,\cdots,N-1]\) and \[\begin{aligned} Y_m = \sum_{n=0}^{N-1} y_n e^{-j 2\pi m n/N}. \end{aligned}\] The inverse discrete Fourier transform (IDFT) reconstructs the original sequence for \(n \in [0,1,\cdots,N-1]\) and \[\begin{aligned} y_n = \frac{1}{N} \sum_{n=0}^{N-1} Y_m e^{j 2\pi m n/N}. \end{aligned}\]
The DFT \((Y_m)\) has a frequency interval equal to the sampling frequency \(\omega_s/N\) and the IDFT \((y_n)\) has time interval equal to the sampling time \(T\). The first \(N/2+1\) DFT \((Y_m)\) values correspond to frequencies \[\begin{aligned} (0,\omega_s/N,2\omega_s/N,\cdots\omega_s/2) \end{aligned}\] and the remaining \(N/2-1\) correspond to frequencies \[\begin{aligned} (-\omega_s/2,-(N-1)\omega_s/N,\cdots,-\omega_s/N). \end{aligned}\]
In practice, the definitions of the DFT and IDFT are not the most
efficent methods of computation. A clever algorithm called the fast
Fourier transform (FFT) computes the DFT much more efficiently.
Although it is a good exercise to roll our own FFT, in this lecture we
will use scipy
’s built-in FFT
algorithm, loaded with the following command.
from scipy import fft
Now, given a time series array y
representing \((y_i)\), the DFT (using
the FFT algorithm) can be computed with the following command.
fft(y)
In the following example, we will apply this method of computing the DFT.
We would like to compute the DFT of a sample sequence (yn) generated by sampling a spaced-out sawtooth. Let’s first generate the sample sequence and plot it.
In addition to scipy
, let’s
import matplotlib
for figures and
numpy
for numerical
computation.
import matplotlib.pyplot as plt
import numpy as np
We define several “control” quantities for the spaced-sawtooth signal.
= 48 # frequency of the signal
f_signal = 1 # spaces between sawteeth
spaces = 10 # number of signal periods
n_periods = 10 # samples/sawtooth n_samples_sawtooth
These quantities imply several “derived” quantities that follow.
= n_samples_sawtooth*(1+spaces)
n_samples_period = n_periods*n_samples_period
n_samples = 1.0/f_signal # period of signal
T_signal = np.linspace(0,n_periods*T_signal,n_samples)
t_a = n_periods*T_signal/(n_samples-1) # sample time
dt = 1./dt # sample frequency f_sample
We want an interval of ramp followed by an interval of “space”
(zeros). The following method of generating the sampled signal y
helps us avoid leakage, which
we’ll describe at the end of the example.
= np.zeros(n_samples_sawtooth) # frac of period
arr_zeros = np.arange(n_samples_sawtooth) # frac of period
arr_ramp = [] # initialize time sequence
y for i in range(n_periods):
= np.append(y,arr_ramp) # ramp
y for j in range(spaces):
= np.append(y,arr_zeros) # space y
We plot the result in , generated by the following code.
= plt.subplots()
fig, ax 'b-',linewidth=2)
plt.plot(t_a,y,'time (s)')
plt.xlabel('$y_n$')
plt.ylabel( plt.show()
Now we have a nice time sequence on which we can perform our DFT. It’s easy enough to compute the FFT.
= fft(y)/n_samples # FFT with proper normalization Y
Recall that the latter values correspond to negative frequencies. In
order to plot it, we want to rearrange our Y
array such that the elements
corresponding to negative frequencies are first. It’s a bit annoying,
but c’est la vie.
= Y[range(int(n_samples/2))]
Y_positive_zero = np.flip(
Y_negative 0),0
np.delete(Y_positive_zero,
)= np.append(Y_negative,Y_positive_zero) Y_total
Now all we need is a corresponding frequency array.
= np.arange(
freq_total -n_samples/2+1,n_samples/2
*f_sample/n_samples )
The plot, created with the following code, is shown in .
= plt.subplots()
fig, ax abs(Y_total),'r-',linewidth=2)
plt.plot(freq_total, 'frequency $f$ (Hz)')
plt.xlabel('$Y_m$')
plt.ylabel( plt.show()
Leakage
The DFT assumes the sequence (yn) is periodic with period N. An implication of this is that if any periodic components have period Nshort < N, unless N is divisible by Nshort, spurious components will appear in (Yn). Avoiding leakage is difficult, in practice. Instead, typically we use a window function to mitigate its effects. Effectively, windowing functions—such as the Bartlett, Hanning, and Hamming windows—multiply (yn) by a function that tapers to zero near the edges of the sample sequence.
Numpy has several
window functions such as bartlett()
, hanning()
, and hamming()
.
Let’s plot the windows to get a feel for them – see .
= np.bartlett(n_samples)
bartlett_window = np.hanning(n_samples)
hanning_window = np.hamming(n_samples)
hamming_window
= plt.subplots()
fig, ax
plt.plot(t_a,bartlett_window,'b-',label='Bartlett',linewidth=2)
plt.plot(t_a,hanning_window,'r-',label='Hanning',linewidth=2)
plt.plot(t_a,hamming_window,'g-',label='Hamming',linewidth=2)
'time (s)')
plt.xlabel('window $w_n$')
plt.ylabel(
plt.legend() plt.show()
Online Resources for Section 7.5
No online resources.