Introduction to functions and integration#
Content modified under Creative Commons Attribution license CC-BY 4.0, code under BSD 3-Clause License © 2020 R.C. Cooper
Building our own functions.#
Functions are saved in memory you can define them with anonymous functions lambda
and function definitions def
Example of storing a function in memory
import numpy as np
import matplotlib.pyplot as plt
plt.style.use('fivethirtyeight')
f= lambda x,y: (x*y**3-x**3*y)
f(1,0.1)
-0.099
The function, f(x, y)
=
linspace
to define the x-y-axesmeshgrid
to fill in each x-y value along each axis, (-1, -1), (-1, -0.99), …(0.99, 1), (1, 1)contourf
to display the result
x=np.linspace(-1,1,101); # 10 data points from -1 to 1
y=np.linspace(-1,1,101); # 10 data points from -1 to 1
X,Y=np.meshgrid(x,y); # 100 data points from -1 to 1 on the x- and y-axes
Z=f(X,Y);# Z=f(X,Y) evaluated at the 100 data points
plt.contourf(X,Y,Z) # also pcolor, countour, ...
<matplotlib.contour.QuadContourSet at 0x79138c00ecb0>
For more plotting examples and options check out the Matplotlib gallery
Here, define a function called my_ode
for a single-DOF forced spring-mass-damper:
Its initial postion and velocity are
In order to use a computational method to integrate this differential equation, you have to define a set of first order differential equations. You set these up as such
Now, you have 2 first-order equations that relate
def my_ode(t, r):
""" Help documentation for "my_ode"
input is time, t (s) and r=[position (m); velocity (m/s)] and time
output is dr=[velocity (m/s); acceleration (m/s/s)] at time, t
the ODE is defined by:
a = -2*v-9*x+ 0.01*cos(2t)"""
dr=np.zeros(np.size(r))
dr[0] = r[1]
dr[1] = -2*r[1]-9*r[0]+0.01*np.cos(2*t);
return dr
Given any [position, velocity]
and time
, the defined my_ode
returns the velocity and acceleration:
my_ode(0,[1, 2])
array([ 2. , -12.99])
Euler approximation#
The simplest integration routine is the Euler approximation. A first-order Taylor series expansion about the current timestep:
You can integrate the equation in a for
-loop.
for
-loop: Is defined in Python by using the function,for
. You specify the variable,i
, and the values,range(1, len(t))
i = (1, 2, … 100).
Each step in the loop approximates the next position based upon the current velocity and the next velocity based upon the current acceleration.
N = 100
t = np.linspace(0, 6*np.pi/2, N) # (start, stop, total steps)
dt = t[1] - t[0] # time step
x0 = 0.1
v0 = 0
r = np.zeros((len(t),2))
r[0,:] = np.array([x0,v0])
for i in range(1,len(t)):
dr = my_ode(t[i-1],r[i-1,:])
r[i, :] = r[i-1,:] + dr*dt
plt.plot(t,r[:,0])
plt.xlabel('time (s)')
plt.ylabel('position (m)')
Text(0, 0.5, 'position (m)')
Runge-Kutta#
Using solve_ivp
in Python you can use more advanced integration routines, such as Runge-Kutta 5(4), Runge-Kutta 3(2), Adams/BDF method, etc.
Numerical integration algorithms for differential equations:
from scipy.integrate import solve_ivp # import the ordinary differential equation integrator in Python
r23=solve_ivp(my_ode,[t[0],t[-1]],[x0, v0],method='RK23');
r45=solve_ivp(my_ode,[t[0],t[-1]],[x0, v0],method='RK45'); # default = 'RK45'
plt.plot(r23.t,r23.y[0],'o',label='RK45')
plt.plot(r45.t,r45.y[0],'s',label='RK23')
plt.plot(t,r[:,0],label='Euler')
plt.legend(loc='center left', bbox_to_anchor=(1, 0.5))
#axis([0 10 -0.2 0.3])
plt.xlabel('time (s)')
plt.ylabel('position (m)')
Text(0, 0.5, 'position (m)')
Freefall Model Computational solution#
Here is our first computational mechanics model.
An object falling is subject to the force of
gravity (
=mg) anddrag (
)
Acceleration of the object:
Define constants and analytical solution (meters-kilogram-sec)#
We define parameters for our problem as the acceleration due to gravity, g, drag coefficient, c, and mass of the object, m. Once we have defined these parameters, we have a single variable whose derivative
parameters:
g=9.81 m/s
function:
We can solve for the analytical solution in this case. First, consider the speed of the falling object when acceleration is
Now, substitute this terminal velocity into the equation and integrate to get the analytical solution v(t):
Exercise:#
Calculate the terminal velocity for the given parameters, g=9.81 m/s
c=0.25
m=60
g=9.81
def v_analytical(t, m = m, g = g, c = c):
'''Analytical solution for the velocity of an object released from rest subject to
the force of gravity and the force of drag with drag coefficient, c
Arguments
---------
t: time, the independent variable
m: mass of the object
g: acceleration due to gravity
c: drag coefficient
Returns
-------
v: the speed of the object at time t'''
v_terminal=np.sqrt(m*g/c)
v= v_terminal*np.tanh(g*t/v_terminal)
return v
Inside the curly brackets—the placeholders for the values we want to print—the f
is for float
and the .4
is for four digits after the decimal dot. The colon here marks the beginning of the format specification (as there are options that can be passed before). There are so many tricks to Python’s string formatter that you’ll usually look up just what you need.
Another useful resource for string formatting is the Python String Format Cookbook. Check it out!
Printing these values using the string formatter, with a total length of 5
and only printing 2 decimal digits, we can display our solution in a human-readable way.
{:5.2f}
where
:5
prints something with whitespace that is 5 spaces total.2
prints 2 significant figures after the decimalf
tellsformat
that the input is a floating point number to print
Analytical vs Computational Solution#
The analytical solution above gives us an exact function for t
, and calculate the speed, v
.
In many engineering problems, we cannot find or may not need an exact mathematical formula for our design process. It is always helpful to compare a computational solution to an analytical solution, because it will tell us if our computational solution is correct. Next, we will develop the Euler approximation to solve the same problem.
Define numerical method#
Finite difference approximation#
Computational models do not solve for functions e.g. v(t), but rather functions at given points in time (or space). In the given freefall example, we can approximate the derivative of speed, the speed at the next poitn in time given the current time and its current speed as such
c=0.25
m=60
g=9.81
def freefall_acceleration(t, v, m = m, g = g, c = c):
'''Analytical solution for the velocity of an object released from rest subject to
the force of gravity and the force of drag with drag coefficient, c
Arguments
---------
t: time, the independent variable
m: mass of the object
g: acceleration due to gravity
c: drag coefficient
Returns
-------
a: the acceleratio of the object at time t given speed v'''
a = g - c/m*v**2
return a
Compare solutions (plotting)#
We can compare solutions in a figure in a number of ways:
plot the values, e.g.
andplot the difference between the values (the absolute error) e.g.
plot the ratio of the values e.g.
(useful in finding bugs, unit analysis, etc.)plot the ratio of the error to the best estimate (the relative error) e.g.
Let’s start with method (1) to compare our analytical and computational solutions.
Import pyplot
and update the default plotting parameters.
t = np.linspace(0, 6, 7)
v_numerical = solve_ivp(freefall_acceleration,
[0, t.max()], # start time = 0s end time = 6 s
np.array([0]), # initial speed is 0 m/s
t_eval = t) # evaluate at timesteps in t
v_numerical
message: The solver successfully reached the end of the integration interval.
success: True
status: 0
t: [ 0.000e+00 1.000e+00 2.000e+00 3.000e+00 4.000e+00
5.000e+00 6.000e+00]
y: [[ 0.000e+00 9.678e+00 1.861e+01 2.628e+01 3.247e+01
3.718e+01 4.064e+01]]
sol: None
t_events: None
y_events: None
nfev: 44
njev: 0
nlu: 0
plt.plot(t,v_analytical(t),'-',label='analytical')
plt.plot(v_numerical.t ,v_numerical.y[0],'o-',label='numerical')
plt.legend()
plt.xlabel('time (s)')
plt.ylabel('velocity (m/s)')
Text(0, 0.5, 'velocity (m/s)')
Note: In the above plot, the numerical solution is given at discrete points connected by lines, while the analytical solution is drawn as a line. This is a helpful convention. We plot discrete data such as numerical solutions or measured data as points and lines while analytical solutions are drawn as lines.
plt.plot(v_numerical.t ,
v_numerical.y[0] - v_analytical(t),
'o-',label='numerical')
plt.xlabel('time (s)')
plt.ylabel('velocity (m/s)')
plt.title('error in numerical result')
Text(0.5, 1.0, 'error in numerical result')
Note: In the above plot, the error is calculated as a difference between numerical and analytical result. Try changing the y axis to a relative error using the formula
What you’ve learned#
Numerical integration with the Euler approximation and
solve_ivp
How to time a numerical solution or a function
How to compare solutions