Author: Shea Smith Email: SerShea.Smith@gmail.com
High level explanation:
The Fourier series was initiaally created to provide a solution to the heat equation. Joseph Fourier found that he could solve this by superimposing sinusoids on top of one another. This superposition is know as the Fourier Series.
In math, the Fourier series is a peridodic function made up of sinusoids. These sinusoids can also be thought of as rotating vectors in the imaginary plane. When these sinusoids, or vectors are summed and weighted, they can be used to represent other signals and provide solutions to otherwise difficult functions.
The emphasis here is clarity for future students learning the topic:
We are interested often in approximating continuous functions on L_2.
The idea behind the generalized Fourier series comes from the definition of a complete basis. Definition (Complete Basis) An orthonormal set {pi, i = 1, . . . , ∞} in a Hilbert space S is a complete basis or total basis if ∀x ∈ S
Note that if: x=∞∑i=1<ci,pi>and<pi,pj>=δij
From the definition above, we can The generalized Fourier Series can be written as: x=∞∑i=1<x,pi>pi
As a familiar example, we often see continuous time signal or functions in L_2[0,T] written as a summation: f(t)=∞∑n=−∞cn1Tej(2πT)nt
From this final equation we can solve for all constants in the generalized fourier series. With these constants the function f(t) can be accuratly represented with the scaled fourier series.
This is a simple visual to how an incresing sum of sinusoids approximates a step response. In this case, the summation takes the form of:
y(t)=−4π(cos(1πt)1−cos(3πt)3+cos(5πt)5−...)As this summation approches infinity, it models the step response.
%matplotlib inline
from __future__ import print_function
from ipywidgets import interact, interactive, fixed, interact_manual
import ipywidgets as widgets
import numpy as np
import matplotlib.pyplot as plt
from numpy import pi
prec = 1000
x = np.arange(0,1,1/prec)
y = np.ones(prec)
y[:int(prec/2)] = -1
def f(n):
xf = np.arange(0,1,1/prec)
yf = np.zeros(prec)
kk=0
Y=0
for v in x:
fp = -4/pi
for k in range(1,n+1,2):
yf[kk] = yf[kk] + fp*np.cos(k*pi*v) / k
fp = fp*-1
kk = kk +1
plt.plot(x,y,'r.',label='impulse')
plt.plot(xf,yf,label='fourier approx.')
plt.legend()
print("Here we see an example of a fourier series approximating an impulse function.")
print("As the number of summed sinusoids in the series increases the approximation becomes better. ")
interact(f, n=widgets.IntSlider(min=1, max=128, step=2, value=1))
Here we see an example of a fourier series approximating an impulse function. As the number of summed sinusoids in the series increases the approximation becomes better.
interactive(children=(IntSlider(value=1, description='n', max=128, min=1, step=2), Output()), _dom_classes=('w…
<function __main__.f(n)>
The example above was a simpler model that was a summation of cosines, but what about a solution that works for all functions? This example employs the generalized form of the Fourier series that can be used to approximate continuous function in L_2. Try running the code below and see how many iterations it would take to approximate sinusoids to the example function:
g(t)=20e−t/2The generalized fourier series can even track noisy functions, try adjusting the noise level below as well for an added depth to the simulation. You can even change the function listed below to approximate your own function.
from __future__ import print_function
from ipywidgets import interact, interactive, fixed, interact_manual
import ipywidgets as widgets
import numpy as np
import matplotlib.pyplot as plt
from numpy import pi
def f(n,noise_level):
prec = 100
nl = noise_level
#n = 10
T = pi
t = np.arange(0,T, T/prec)
noise = np.random.normal(0,nl,prec)
###Change g_t to your own function!
g_t = 20*np.exp(-t/2)
###
g_t = g_t + noise
f_t = np.zeros(prec)
for k in range(n):
b = np.exp(-1j*k*t*(2*pi/T)) # k = n
c = np.inner(g_t,np.conj(b)*(1/(T)))
f_tc = f_t
f_t = f_t + c*b
plt.plot(t,g_t/np.linalg.norm(g_t),label='noisy function')
plt.plot(t,f_t.real/np.linalg.norm(f_t),label='generalized approx')
plt.legend()
interact(f, n=widgets.IntSlider(min=2, max=1000, step=10, value=2),noise_level=widgets.FloatSlider(min=.00, max=.5, step=.05, value=.00))
interactive(children=(IntSlider(value=2, description='n', max=1000, min=2, step=10), FloatSlider(value=0.0, de…
<function __main__.f(n, noise_level)>
Add a homework assignment that might take 10 minutes to complete. Make sure you can work the problem yourself, but you do not need to submit a solution to the problem.