Steady states in Non-Linear Coupled ODE’s
Contents
Steady states in Non-Linear Coupled ODE’s#
Let’s start with the same predator/prey example from last week.
with \(\vec{y}=[x,y], \vec{y}(t=0)=[1,5]\), and \(\alpha=1, \beta=0.2, \delta=0.5, \gamma=0.2\).
Let’s first plot the solution again for this system starting from \(\vec{y}(t=0)=[1,5]\)
from scipy.integrate import solve_ivp
import numpy as np
import matplotlib.pyplot as plt
alpha = 1 # 1/day
beta = 0.2 #1/wolves/day
delta = 0.5 #1/rabbits/day
gamma = 0.2 #1/day
def diff_eq(t,population):
# t is independent variable
# y is a vector of things we want solve_ivp to integrate for
# y = [y1, y2]
# y = [x,y]
# return vector the same shape as population/y
# this will be the differential equation for each thing in population/y
x,y = population
return [alpha*x - beta*x*y,
delta*x*y - gamma*y]
t_span = [0, 100]
initial_population = [0.5, 5.1]
t_eval = np.linspace(0,100, 500)
#Solve the differential equation
sol = solve_ivp(diff_eq,
t_span,
initial_population,
t_eval=t_eval)
#Plot the solution vs time
plt.plot(sol.t, sol.y.T)
plt.xlabel('Time [days]')
plt.ylabel('Population')
plt.legend(['Rabbits','Wolves'])
plt.show()
# Plot the phase portrait and grid
plt.plot(sol.y[0,:], sol.y[1,:])
plt.plot(sol.y[0,0:1], sol.y[1,0:1],'or')
plt.xlabel('Population of Rabbits')
plt.ylabel('Population of Wolves')
#Plot the phase portrait grid
r = np.linspace(0, 1.5, 20) # rabbit grid
f = np.linspace(0, 10, 20) # wolves grid
R, F = np.meshgrid(r, f) # 2D arrays of (rabbit, wolves) points
DR, DF = diff_eq(0, [R, F])
# This normalizes the arrows so they are all the same length and just show the direction
N = np.sqrt(DR**2 + DF**2)
N[N==0] = 1 # eliminate / 0 errors, it is sort of optional.
DR /= N
DF /= N
plt.quiver(R, F, DR, DF)
<matplotlib.quiver.Quiver at 0x7efbcbcb8eb0>
What is special about that point in the center?
All of the differential equations are 0; it’s a steady state! First, let’s look at this mathematically. This is a nonlinear system, so we have to be careful.
The steady state is defined by \(\vec{y}'=\vec{0}\), so
or
Let’s work out the steady state solutions
So, there are two possibilities:
What happens if we start the solution at either of these steady states?
from scipy.integrate import solve_ivp
import numpy as np
import matplotlib.pyplot as plt
alpha = 1 # 1/day
beta = 0.2 #1/wolves/day
delta = 0.5 #1/rabbits/day
gamma = 0.2 #1/day
def diff_eq(t,population):
# t is independent variable
# y is a vector of things we want solve_ivp to integrate for
# y = [y1, y2]
# y = [x,y]
# return vector the same shape as population/y
# this will be the differential equation for each thing in population/y
x,y = population
return [alpha*x - beta*x*y,
delta*x*y - gamma*y]
t_span = [0, 100]
initial_population = [0.4, 5]
t_eval = np.linspace(0,100, 500)
#Solve the differential equation
sol = solve_ivp(diff_eq,
t_span,
initial_population,
t_eval=t_eval)
#Plot the solution vs time
plt.plot(sol.t, sol.y.T)
plt.xlabel('Time [days]')
plt.ylabel('Population')
plt.legend(['Rabbits','Wolves'])
plt.show()
# Plot the phase portrait and grid
plt.plot(sol.y[0,:], sol.y[1,:])
plt.plot(sol.y[0,0:1], sol.y[1,0:1],'or')
plt.xlabel('Population of Rabbits')
plt.ylabel('Population of Wolves')
#Plot the phase portrait grid
r = np.linspace(0, 1.5, 20) # rabbit grid
f = np.linspace(0, 10, 20) # wolves grid
R, F = np.meshgrid(r, f) # 2D arrays of (rabbit, wolves) points
DR, DF = diff_eq(0, [R, F])
# This normalizes the arrows so they are all the same length and just show the direction
N = np.sqrt(DR**2 + DF**2)
N[N==0] = 1 # eliminate / 0 errors, it is sort of optional.
DR /= N
DF /= N
plt.quiver(R, F, DR, DF)
<matplotlib.quiver.Quiver at 0x7efbcbb86ef0>
from scipy.integrate import solve_ivp
import numpy as np
import matplotlib.pyplot as plt
alpha = 1 # 1/day
beta = 0.2 #1/wolves/day
delta = 0.5 #1/rabbits/day
gamma = 0.2 #1/day
def diff_eq(t,population):
# t is independent variable
# y is a vector of things we want solve_ivp to integrate for
# y = [y1, y2]
# y = [x,y]
# return vector the same shape as population/y
# this will be the differential equation for each thing in population/y
x,y = population
return [alpha*x - beta*x*y,
delta*x*y - gamma*y]
t_span = [0, 100]
initial_population = [0, 0]
t_eval = np.linspace(0,100, 500)
#Solve the differential equation
sol = solve_ivp(diff_eq,
t_span,
initial_population,
t_eval=t_eval)
#Plot the solution vs time
plt.plot(sol.t, sol.y.T)
plt.xlabel('Time [days]')
plt.ylabel('Population')
plt.legend(['Rabbits','Wolves'])
plt.show()
# Plot the phase portrait and grid
plt.plot(sol.y[0,:], sol.y[1,:])
plt.plot(sol.y[0,0:1], sol.y[1,0:1],'or')
plt.xlabel('Population of Rabbits')
plt.ylabel('Population of Wolves')
#Plot the phase portrait grid
r = np.linspace(0, 1.5, 20) # rabbit grid
f = np.linspace(0, 10, 20) # wolves grid
R, F = np.meshgrid(r, f) # 2D arrays of (rabbit, wolves) points
DR, DF = diff_eq(0, [R, F])
# This normalizes the arrows so they are all the same length and just show the direction
N = np.sqrt(DR**2 + DF**2)
N[N==0] = 1 # eliminate / 0 errors, it is sort of optional.
DR /= N
DF /= N
plt.quiver(R, F, DR, DF)
<matplotlib.quiver.Quiver at 0x7efbc8ad8d30>
Perfectly steady!
Now, let’s see what happens if we are close the steady state, but not perfect. Try adjusting the initial condition and see how this behaves.
from scipy.integrate import solve_ivp
import numpy as np
import matplotlib.pyplot as plt
alpha = 1 # 1/day
beta = 0.2 #1/wolves/day
delta = 0.5 #1/rabbits/day
gamma = 0.2 #1/day
def diff_eq(t,population):
# t is independent variable
# y is a vector of things we want solve_ivp to integrate for
# y = [y1, y2]
# y = [x,y]
# return vector the same shape as population/y
# this will be the differential equation for each thing in population/y
x,y = population
return [alpha*x - beta*x*y,
delta*x*y - gamma*y]
t_span = [0, 150]
initial_population = [0.01, 0.01]
t_eval = np.linspace(0,150, 500)
#Solve the differential equation
sol = solve_ivp(diff_eq,
t_span,
initial_population,
t_eval=t_eval,
rtol=1e-10,
atol=1e-12)
#Plot the solution vs time
plt.plot(sol.t, sol.y.T[:,1])
plt.xlabel('Time [days]')
plt.ylabel('Population')
plt.legend(['Rabbits','Wolves'])
plt.show()
# Plot the phase portrait and grid
plt.plot(sol.y[0,:], sol.y[1,:])
plt.plot(sol.y[0,0:1], sol.y[1,0:1],'or')
plt.xlabel('Population of Rabbits')
plt.ylabel('Population of Wolves')
#Plot the phase portrait grid
r = np.linspace(0, 1.5, 20) # rabbit grid
f = np.linspace(0, 10, 20) # wolves grid
R, F = np.meshgrid(r, f) # 2D arrays of (rabbit, wolves) points
DR, DF = diff_eq(0, [R, F])
# This normalizes the arrows so they are all the same length and just show the direction
N = np.sqrt(DR**2 + DF**2)
N[N==0] = 1 # eliminate / 0 errors, it is sort of optional.
DR /= N
DF /= N
plt.quiver(R, F, DR, DF)
<matplotlib.quiver.Quiver at 0x7efbc898fd00>
It’s hard to see, but even here we are facing some numerical problems. Try changing the tolerances in your code and see what happens.
The only reason I thought this was suspicious because of the analysis I did later to understand how this should behave around the steady state!
Try plotting just the wolf population and try different initial conditions. Notice anything about what the solution looks like?
Numerical calculation of steady states#
We can use numerical methods to find steady states in our system. This is a specific case of a general problem of finding a vector \(\vec{x}\) such that some function
where \(\vec{0}\) is the same length as \(\vec{x}\). These methods work well, but they have a couple downsides:
They are numerical so you have to be a little wary of solutions due to numerical problems
You have to provide an initial guess for the steady state
The method will return one solution, but not every solution. It can not tell you how many possible solutions there are.
The only tricky thing is that we have to change our function.
# We need this to work like fun(x)=0
def diff_eq(t,population):
# t is independent variable
# y is a vector of things we want solve_ivp to integrate for
# y = [y1, y2]
# y = [x,y]
# return vector the same shape as population/y
# this will be the differential equation for each thing in population/y
x,y = population
return [alpha*x - beta*x*y,
delta*x*y - gamma*y]
def myfun(x):
return 2*x
print(myfun(1))
myfun = lambda x: 2*x # is shorthand for def myfun(x): return x
print(myfun(1))
new_diff_eq = lambda x: diff_eq(0,x)
# # example with two inputs
# def myfun(x, y):
# return 2*x
# myfun = lambda x,y: 2*x # is shorthand for def myfun(x): return x
# print(myfun(1, 2))
2
2
To read more about lambda functions: link texthttps://www.w3schools.com/python/python_lambda.asp
Ok, now we can use root with our existing diff_eq function!
from scipy.optimize import fsolve
def diff_eq(t, population):
# t is independent variable
# y is a vector of things we want solve_ivp to integrate for
# y = [y1, y2]
# y = [x,y]
# return vector the same shape as population/y
# this will be the differential equation for each thing in population/y
x,y = population
return [alpha*x - beta*x*y,
delta*x*y - gamma*y]
solution = fsolve(diff_eq,
[0.5, 0.5])
solution
---------------------------------------------------------------------------
TypeError Traceback (most recent call last)
Cell In [7], line 16
11 x,y = population
13 return [alpha*x - beta*x*y,
14 delta*x*y - gamma*y]
---> 16 solution = fsolve(diff_eq,
17 [0.5, 0.5])
19 solution
File ~/micromamba-root/envs/buildenv/lib/python3.10/site-packages/scipy/optimize/_minpack_py.py:160, in fsolve(func, x0, args, fprime, full_output, col_deriv, xtol, maxfev, band, epsfcn, factor, diag)
49 """
50 Find the roots of a function.
51
(...)
150
151 """
152 options = {'col_deriv': col_deriv,
153 'xtol': xtol,
154 'maxfev': maxfev,
(...)
157 'factor': factor,
158 'diag': diag}
--> 160 res = _root_hybr(func, x0, args, jac=fprime, **options)
161 if full_output:
162 x = res['x']
File ~/micromamba-root/envs/buildenv/lib/python3.10/site-packages/scipy/optimize/_minpack_py.py:226, in _root_hybr(func, x0, args, jac, col_deriv, xtol, maxfev, band, eps, factor, diag, **unknown_options)
224 if not isinstance(args, tuple):
225 args = (args,)
--> 226 shape, dtype = _check_func('fsolve', 'func', func, x0, args, n, (n,))
227 if epsfcn is None:
228 epsfcn = finfo(dtype).eps
File ~/micromamba-root/envs/buildenv/lib/python3.10/site-packages/scipy/optimize/_minpack_py.py:24, in _check_func(checker, argname, thefunc, x0, args, numinputs, output_shape)
22 def _check_func(checker, argname, thefunc, x0, args, numinputs,
23 output_shape=None):
---> 24 res = atleast_1d(thefunc(*((x0[:numinputs],) + args)))
25 if (output_shape is not None) and (shape(res) != output_shape):
26 if (output_shape[0] != 1):
TypeError: diff_eq() missing 1 required positional argument: 'population'
Stability of Steady States in Nonlinear Systems#
Steady states defined by
For linear homogeneous systems, this is straightforward
\(\underline{\text{Ex}}\)
Only solution: \(\vec{y}^{ss} = \vec{0}\)
There is a steady state, but \(\lim_{t \to \infty} \vec{y} \neq \vec{0}\), \(\lambda\) are complex. Only steady state by starting it at \(\vec{y} = \vec{0}\)
Stability in a 1d system#
Critical points for a function \(y' = 0\). Multiple types of critical points depending on \(y''\)
Stability for ODE’s#
Same idea applied for coupled ODE’s. Eigenvalues are equivalent for linear system. We’re going to come up with Taylor series approximations for \(y_1'\) and \(y_2'\) that is valid near the steady state. \(\vec{y}_{ss}\) See https://en.wikipedia.org/wiki/Taylor_series
At the steady state, \(\vec{y}' = \begin{bmatrix} y_{1,ss}' \\ y_{2,ss}' \end{bmatrix} = \begin{bmatrix} y_1' \\ y_2' \end{bmatrix}_{y_1=y_{1,ss},\ y_2 = y_{2,ss}} = \begin{bmatrix} 0 \\ 0 \end{bmatrix}\)
So,
Call the first matrix the Jacobian. The Jacobian is a matrix of all of the possible second derivatives in the problem.
This is a linear approximation that holds near a steady state solution.
Define the special matrix, the Jacobian
Two great tables from the textbook (4.1,4.2):
![table_41.png]()
![table_42.png]()
\(\underline{\text{Ex}}\):
For a linear system, the Jacobian of the ODE is the same as \(\arr{A}\)
The linearization of a linear function is just the linear function taylor series of \(1 + x = 1 + x\)
\(\underline{\text{Ex}} 2\):
Get steady states
3 solutions: \(\vec{y}^{ss} = \begin{bmatrix} 0\\0 \end{bmatrix}, \begin{bmatrix} 3 \\ 9 \end{bmatrix}, \begin{bmatrix} -2 \\ 4 \end{bmatrix}\)
2. Calculate Jacobian:
The Jacobian tells us about the stability at the steady state.
Calculate the Jacobian at each steady state.
For the first steady state [0,0]
Infinitely close to \(\vec{y} = \begin{bmatrix} 0 \\ 0 \end{bmatrix}\), this system is well modeled by \(\vec{y}' = \arr{A}\vec{y} = \begin{bmatrix} -6 & -1 \\ 0 & 1 \end{bmatrix} \vec{y}\)
Use our knowledge about stability of steady state systems:
For the second steady state [3,9]
Infinitely close to \(\vec{y} = \begin{bmatrix} 3 \\ 9 \end{bmatrix}\), this system is well modeled by \(\vec{y}' = \arr{A}\vec{y} =\begin{bmatrix} 3&2\\-6&1 \end{bmatrix} \vec{y}\)
Let’s do this one numerically
import numpy as np
A = [[3,2],
[-6,1]]
eigval, eigvec = np.linalg.eig(A)
print(eigval)
[2.+3.31662479j 2.-3.31662479j]
Solution for \(\vec{y} = \begin{bmatrix} 3 \\ 9 \end{bmatrix}\)
This says the solution will exponentially increase, but oscillate as it does so (like \(e^{2t}\cos 3.3t\)) (move away from S.S.).
So the eigenvalues for the second steady state are \(2+3.3i, 2-3.3i\).
One last steady state! [-2, 4]
Final: \(\vec{y}^{ss} = \begin{bmatrix} -2 \\ 4 \end{bmatrix}\)
Will oscillate but oscillations will decay over time
import numpy as np
A = [[-2,-3],
[4,1]]
eigval, eigvec = np.linalg.eig(A)
print(eigval)
[-0.5+3.122499j -0.5-3.122499j]
This will oscillate but exponentially decrease, like \(e^{-t/2}\cos 3.12t\) . Let’s evaluate the solution now.
In-class exercise - plot the numerical solution starting from [1,1]#
from scipy.integrate import solve_ivp
import numpy as np
import matplotlib.pyplot as plt
def diff_eq(t,y):
y1, y2 = y
return [y1*y2-y2-6*y1,
-y1**2+y2]
t_span = [0, 10]
y0 = [3,9]
t_eval = np.linspace(0,10, 100)
#Solve the differential equation
sol = solve_ivp(diff_eq,
t_span,
y0,
t_eval=t_eval)
#Plot the solution vs time
plt.plot(sol.t, sol.y.T)
plt.xlabel('t')
plt.legend(['y_1','y_2'])
plt.show()
# Plot the phase portrait
plt.plot(sol.y[0,:], sol.y[1,:])
plt.plot(sol.y[0,0:1], sol.y[1,0:1],'or')
plt.xlabel('y_1')
plt.ylabel('y_2')
#Plot the phase portrait grid
r = np.linspace(-7, 10, 20) # y1 grid
f = np.linspace(-7, 20, 20) # y2 grid
R, F = np.meshgrid(r, f) # 2D arrays of (rabbit, wolves) points
DR, DF = diff_eq(0, [R, F])
# This normalizes the arrows so they are all the same length and just show the direction
N = np.sqrt(DR**2 + DF**2)
N[N==0] = 1 # eliminate / 0 errors, it is sort of optional.
DR /= N
DF /= N
plt.quiver(R, F, DR, DF)
<matplotlib.quiver.Quiver at 0x7f4c2bb42390>
Try playing around with differential initial conditions and see how the system behaves. Try adding points for the steady states on the secnod plot.
Recap: We can’t solve nonlinear ODE’s but we can say something quantitative about steady state solutions and their stability!#
Coupled nonlinear ODE’s:
Solve for steady solutions (\(\vec{y}'=\vec{0}\)). Be careful, there may be 0-\(\infty\) solutions!
Find the Jacobian of the coupled ODE’s
For each steady state, evaluate Jacobian and calculate eigenvalues
Classify stability:
If real part of eigenvalues all negative \(\rightarrow\) stable steady state
If some real parts are positive, probably unstable
If complex, may be some oscillations around steady state
If only imaginary, stable oscillations
We can solve these numerically as well:
Use
solve_ivp
to integrate from a set of initial conditionsBe careful about tolerances, by default rtol=1e-3, and atol=1e-6
Try smaller (rtol=1e-6, atol=1e-8). If the answer changes, you have a problem. Decrease the rtol/atol until your answer no longer changes with those values
We can find steady states of the system numerically using
root
If you want to use the same differential equation function, you need to use lambda function or similar idea to handle the
t
input.root
requires an initial guess, and it will yield A solution, not every solution. It is very dependent on the initial guess. There are no guarantees or ways to find all solutions without using special methods from Process Systems Engineering research.
Return to the wolf/rabbit#
Wolf/rabbit example
Solve for S.S.
Two S.S.:
Jacobian
Evaluate Jacobian at each S.S.
At \(x=0, y=0\)
Only 1 negative eigenvalue, not stable, unless you approach from a certain direction! At \(x = 0.4, y = 5\):
Stable oscillations in population of wolf/rabbits
J = [[0,-0.08],
[2.5,0]]
np.linalg.eig(J)
(array([0.+0.4472136j, 0.-0.4472136j]),
array([[0. +0.17609018j, 0. -0.17609018j],
[0.98437404+0.j , 0.98437404-0.j ]]))
Numerical estimate of the jacobian#
First, let’s find the steady state again numerically.
from scipy.optimize import root
alpha = 1 # 1/day
beta = 0.2 #1/wolves/day
delta = 0.5 #1/rabbits/day
gamma = 0.2 #1/day
def diff_eq(t,population):
# t is independent variable
# y is a vector of things we want solve_ivp to integrate for
# y = [y1, y2]
# y = [x,y]
# return vector the same shape as population/y
# this will be the differential equation for each thing in population/y
x,y = population
return np.array([alpha*x - beta*x*y,
delta*x*y - gamma*y])
# find a steady state starting from [2, 5]
steady_state = fsolve(lambda x: diff_eq(0, x),
[2,5])
print('I guessed %s, and found a steady state %s' % ([2,5],steady_state))
I guessed [2, 5], and found a steady state [0.4 5. ]
We can get a numerical estimate of the jacobian, but it requires another package numdifftools
.
! pip install numdifftools
import numdifftools as nd
jac_fun = nd.Jacobian(lambda x: diff_eq(0, x))
jac = jac_fun(steady_state)
np.linalg.eig(jac)
Requirement already satisfied: numdifftools in /usr/local/lib/python3.7/dist-packages (0.9.39)
(array([3.92290495e-18+0.4472136j, 3.92290495e-18-0.4472136j]),
array([[1.54464231e-18+0.17609018j, 1.54464231e-18-0.17609018j],
[9.84374039e-01+0.j , 9.84374039e-01-0.j ]]))
jac
array([[ 7.84580989e-18, -8.00000000e-02],
[ 2.50000000e+00, 0.00000000e+00]])