Solving Ordinary Differential Equations
Engineering analysis regularly includes the solution of differential equations. Differential equations are those equations that contain derivatives. An ordinary differential equation (ODE) is a differential equation that contains only ordinary, as opposed to partial, derivatives. A linear ODE—one for which constant multiples and sums of solutions are also solutions—is an important type that represent linear, time-varying (LTV) systems. For this class of ODEs, it has been proven that for a set of initial conditions, a unique solution exists (Kreyszig 2010, 108).
A constant-coefficient, linear ODE can represent linear, time-invariant (LTI) systems. An LTV or LTI system model can be represented as a scalar \(n\)th-order ODE, or as a system of \(n\) 1st-order ODEs. As a scalar \(n\)th-order linear ODE, with independent time variable \(t\), output function \(y(t)\), forcing function \(f(t)\), and constant coefficients \(a_i\), has the form \[ y^{(n)}(t) + a_{n-1} y^{(n-1)}(t) + \cdots a_1 y'(t) + a_0 y(t) = f(t). \qquad{(1)}\] The forcing function \(f(t)\) can be written as a linear combination of derivatives of the input function \(u(t)\) with \(m+1 \le n+1\) constant coefficients \(b_j\), as follows: \[ f(t) = b_{m} u^{(m)}(t) + b_{m-1} u^{(m-1)}(t) + \cdots + b_1 u'(t) + b_0 u(t). \] Alternatively, the same LTI system model can be represented by a system of \(n\) 1st-order ODEs, which can be written in vector form as $$\begin{subequations}\label{eq:state-space-model} \begin{align} \vec{x}'(t) &= \mat{A} \vec{x}(t) + \mat{B} \vec{u}(t)\label{eq:state-space-model-state} \\ \vec{y}(t) &= \mat{C} \vec{x}(t) + \mat{D} \vec{u}(t), \label{eq:state-space-model-output} \end{align} \end{subequations}$$ where \(\vec{x}(t)\) is called the state vector, \(\vec{u}(t)\) is called the input vector, and \(\vec{y}(t)\) is called the output vector (they are actually vector-valued functions of time), and \(\mat{A}\), \(\mat{B}\), \(\mat{C}\), and \(\mat{D}\) are matrices containing constants derived from system parameters (e.g., a mass, a spring constant, a capacitance, etc.). Eq. ¿eq:state-space-model? is called an LTI state-space model, and it is used to model a great many engineering systems.
Solving ODEs and systems of ODEs is a major topic of mathematical engineering analysis. It is typically the primary topic of one required course and a secondary topic of several others. Understanding when these solutions exist, whether they are unique, and how they can be found adds much to the understanding of engineering systems. However, it is also true that CASs such as SymPy offer the engineer excellent tools for making quick and adaptable work of this task.
Consider the ODE \[
3 y'(t) + y(t) = f(t),
\] where the forcing function \(f(t)\) is defined piecewise as \[
f(t) = \begin{cases} 0 & t < 0 \\ 1 & t \ge 0. \end{cases}
\] The SymPy dsolve()
function can find the general solution
(i.e., a family of solutions for any initial conditions) with the
following code:
= sp.symbols("t", nonnegative=True)
t = sp.Function("y", real=True)
y = 1 # Or sp.Piecewise(), but $t \ge 0$ already restricts $f(t)$
f = sp.Eq(3*y(t).diff(t) + y(t), f) # Define the ODE
ode = sp.dsolve(ode, y(t)); sol # Solve sol
The solution is returned as an sp.Eq()
equation object. Note the
unknown constant \(C_1\) in the
solution. To find the specific solution
(i.e., the general solution with the initial condition applied to
determine \(C_1\)) for a given initial
condition \(y(0) = 5\),
= sp.dsolve(ode, y(t), ics={y(0): 5}); sol sol
Now consider the ODE \[
y''(t) + 5 y'(t) + 9 y(t) = 0.
\] The SymPy dsolve()
function can find the general solution with the following code:
= sp.Eq(y(t).diff(t, 2) + 5*y(t).diff(t) + 9*y(t), 0)
ode = sp.dsolve(ode, y(t)); sol sol
This is a decaying sinusoid. Applying two initial conditions, \(y(0) = 4\) and \(y'(0) = 0\), we obtain the following:
= sp.dsolve(
sol
ode, y(t), ={y(0): 4, y(t).diff(t).subs(t, 0): 0}
ics; sol )
We see here that to apply the initial condition \(y'(0) = 0\), the derivative must be applied before substituting \(t \rightarrow 0\).
Solving sets (i.e., systems) of first-order differential equations is similar. Consider the set of differential equations \[ y_1'(t) = y_2(t) - y_1(t) \text{ and } y_2'(t) = y_1(t) - y_2(t). \] To find the solution for initial conditions \(y_1(0) = 1\) and \(y_2(0) = -1\), we can use the following technique:
= sp.symbols("t", nonnegative=True)
t = sp.symbols("y1, y2", cls=sp.Function, real=True)
y1, y2 = [y1(t).diff(t) + y1(t) - y2(t), y2(t).diff(t) + y2(t) - y1(t)]
odes = {y1(0): 1, y2(0): -1}
ics = sp.dsolve(odes, [y1(t), y2(t)], ics=ics)
sol print(sol)
[Eq(y1(t), exp(-2*t)), Eq(y2(t), -exp(-2*t))]
In engineering, it is common to express a set of differential equations as a state-space model, as in eq. ¿eq:state-space-model?. The following example demonstrates how to solve these with SymPy.
Consider the electromechanical schematic of a direct current (DC) motor shown in [@fig:motor-circuit]. A voltage source VS(t) provides power, the armature winding loses some energy to heat through a resistance R and stores some energy in a magnetic field due to its inductance L, which arises from its coiled structure. An electromechanical interaction through the magnetic field, shown as M, has torque constant KM and induces a torque on the motor shaft, which is supported by bearings that lose some energy to heat via a damping coefficient B. The rotor’s mass has rotational moment of inertia J, which stores kinetic energy. We denote the voltage across an element with v, the current through an element with i, the angular velocity across an element with Ω, and the torque through an element with T.
A state-space model state equation in the form of [@eq:state-space-model-state] can be derived for this system, with the result as follows:
$$ \underbrace{ \frac{d}{d t} \begin{bmatrix} \Omega_J \\ i_L \end{bmatrix} }_{\vec{x}'(t)} = \underbrace{ \begin{bmatrix} -B/J & K_M/J \\ -K_M/L & -R/L \end{bmatrix} }_{\mat{A}} \underbrace{ \begin{bmatrix} \Omega_J \\ i_L \end{bmatrix} }_{\vec{x}(t)} + \underbrace{ \begin{bmatrix} 0 \\ 1/L \end{bmatrix} }_{\mat{B}} \underbrace{ \begin{bmatrix} V_S \end{bmatrix} \vphantom{\begin{bmatrix}0\\0\end{bmatrix}} }_{\vec{u}(t)}. $$ We choose $\vec{y} = \begin{bmatrix} \Omega_J \end{bmatrix}$ as the output vector, which yields output equation (i.e., [@eq:state-space-model-output]) $$ \underbrace{ \begin{bmatrix} \Omega_J \end{bmatrix} \vphantom{\begin{bmatrix}0\\0\end{bmatrix}} }_{\vec{y}(t)} = \underbrace{ \begin{bmatrix} 1 & 0 \end{bmatrix} \vphantom{\begin{bmatrix}0\\0\end{bmatrix}} }_{\mat{C}} \underbrace{ \begin{bmatrix} \Omega_J \\ i_L \end{bmatrix} }_{\vec{x}(t)} + \underbrace{ \begin{bmatrix} 0 \end{bmatrix} \vphantom{\begin{bmatrix}0\\0\end{bmatrix}} }_{\mat{D}} \underbrace{ \begin{bmatrix} V_S \end{bmatrix} \vphantom{\begin{bmatrix}0\\0\end{bmatrix}} }_{\vec{u}(t)}. $$ Together, these equations are a state-space model for the system.
Solve the state equation for x⃗(t) and the output equation for y⃗(t) for the following case:
- The input voltage VS(t) = 1 V for t ≥ 0
- The initial condition is $\vec{x}(0) = \vec{0}$
We begin by defining the parameters and functions of time as SymPy symbolic variables and unspecified functions as follows:
= sp.symbols("R, L, K_M, B, J", positive=True)
R, L, K_M, B, J = sp.symbols(
W_J, i_L, V_S "W_J, i_L, V_S", cls=sp.Function, real=True
# $\Omega_J, i_L, V_S$
) = sp.symbols("t", real=True) t
Now we can form the symbolic matrices and vectors:
= sp.Matrix([[-B/J, K_M/J], [-K_M/L, -R/L]]) # $\mat{A}$
A_ = sp.Matrix([[0], [1/L]]) # $\mat{B}$
B_ = sp.Matrix([[1, 0]]) # $\mat{C}$
C_ = sp.Matrix([[0]]) # $\mat{D}$
D_ = sp.Matrix([[W_J(t)], [i_L(t)]]) # $\vec{x}$
x = sp.Matrix([[V_S(t)]]) # $\vec{u}$
u = sp.Matrix([[W_J(t)]]) # $\vec{y}$ y
The input and initial conditions can be encoded as follows:
= {V_S(t): 1}
u_subs = {W_J(0): 0, i_L(0): 0} ics
The set of first-order ODEs comprising the state equation can be defined as follows:
= x.diff(t) - A_*x - B_*u
odes print(odes)
= sp.dsolve(list(odes.subs(u_subs)), list(x), ics=ics) x_sol
The symbolic solutions for x(t) are lengthy expressions. Instead of printing them, we will graph them for the following set of parameters:
= {
params 1, # (Ohms)
R: 0.1e-6, # (H)
L: 7, # (N$\cdot$m/A)
K_M: 0.1e-6, # (N$\cdot$m/(rad/s))
B: 2e-6, # (kg$\cdot$m$^2$)
J: }
Create a numerically evaluable version of each function as follows:
= sp.lambdify(t, x_sol[0].rhs.subs(params), modules="numpy")
W_J_ = sp.lambdify(t, x_sol[1].rhs.subs(params), modules="numpy") i_L_
Graph each solution as follows:
= np.linspace(0, 0.000002, 201)
t_ = plt.subplots(2, sharex=True)
fig, axs 0].plot(t_, W_J_(t_))
axs[1].plot(t_, i_L_(t_))
axs[1].set_xlabel("Time (s)")
axs[0].set_ylabel("$\\Omega_J(t)$ (rad/s)")
axs[1].set_ylabel("$i_L(t)$ (A)")
axs[ plt.show()
The output equation is trivial in this case, yielding only the state variable ΩJ(t), for which we have already solved. Therefore, we have completed the analysis.
Online Resources for Section 4.7
No online resources.