A vibration example with two modes
In the following example, we explore the a mechanical vibration example, especially with regard to its modes of vibration. Both undamped and (under)damped cases are considered and we discover the effects of damping.
Consider the system of figure 7.3 in which a velocity source VS is applied to spring K1, which connects to mass m1, which in turn is connected via spring K2 and damper B to mass m2 which.1
The state-space model A-matrix is given as $$\begin{aligned} A &= \begin{bmatrix} -B/m_1 & -1/m_1 & B/m_1 & 0 \\ K_1 & 0 & -K_1 & 0 \\ B/m_2 & 1/m_2 & -B/m_2 & -1/m_2 \\ 0 & 0 & K_2 & 0 \end{bmatrix} \end{aligned}$$ with parameters as follows:
m1 = 0.1 kg
m2 = 1.1 kg
K1 = 8 N/m
K2 = 9 N/m
Two different values for B will be considered: 0 and 20 N⋅s/m. We will explore the modes of vibration in each case and plot a corresponding free response.
This common situation appears in a slightly modified form in @Rowell1997.↩︎
Setting up the problem
We analyze the problem with Python. First, we load packages for symbolic, numeric, and graphical analysis, as follow:
import sympy as sp
import numpy as np
import matplotlib.pyplot as plt
from pprint import pprint
The A matrix is first defined symbolically.
"m_1, m_2, K_1, K_2, B", real=True)
sp.var(= sp.Matrix([
A -B/m_1, -1/m_1, B/m_1, 0],
[0, -K_1, 0],
[K_1, /m_2, 1/m_2, -B/m_2, -1/m_2],
[B0, 0, K_2, 0]
[ ])
Now define dictionaries for the parameter values.
= {
p 0.1, # kg
m_1: 1.1, # kg
m_2: 8, # N/m
K_1: 9 # N/m
K_2:
}= {B: 0} # N/(rad/s), without damping
pB1 = {B: 20} # N/(rad/s), with damping pB2
Without damping
Without damping, we expect the system to be marginally stable and
have two pairs of second-order undamped subsystems with their own unique
natural frequencies. The numerical A matrix can be computed by
substituting in the parameters in p
and pB1
, as follows:
= np.array(A.subs(p).subs(pB1), dtype=float)
A_1 print(A_1)
[[ 0. -10. 0. 0. ]
[ 8. 0. -8. 0. ]
[ 0. 0.90909091 0. -0.90909091]
[ 0. 0. 9. 0. ]]
To explore the modes of vibration, we consider the eigendecomposition of A.
= np.linalg.eig(A_1)
l_,M_ = 1e-14 # threshold for calling something 0
thr abs(l_.real) < thr] = 0.0 # zeroing small real parts l_.real[
Let’s take a closer look at the eigenvalues.
print(l_)
[0.+9.38179379j 0.-9.38179379j 0.+2.726993j 0.-2.726993j ]
So we have two pairs of purely imaginary eigenvalues. We would say, then, that there are two “modes of vibration,” and similarly two second-order systems comprising this fourth-order system. When we consider what the natural frequency and damping ratio is for each pair, we’re considering the natural frequencies associated with each “mode of vibration.”
For a second-order system (see ), the roots of the characteristic equation, which are equal to the eigenvalues corresponding to that second-order pair, are given in terms of natural frequency ωn and damping ratio ζ: $$\maybeeq{ \begin{equation*} -\omega_n\zeta \pm \omega_n \sqrt{\zeta^2-1}. \end{equation*} }$$
So the imaginary part is nonzero only when ζ ∈ [0, 1), that is, when the system is underdamped or undamped. In this case, $$\maybeeq{ \begin{equation} -\omega_n\zeta \pm j \omega_n \sqrt{1-\zeta^2}. \end{equation} }$$
This, taken with the fact that the eigenvalues in l_
have zero real parts, implies either
ωn or
ζ is zero. But if ωn is zero, the
eigenvalues would all be zero, which they are not. Therefore, ζ = 0 for both pairs of
eigenvalues.
This leaves us with eigenvalues: ± jωn1 and ± jωn2.
So we can easily identify the natural frequencies ωn1 and ωn2 associated with each mode as follows.
= np.imag(l_[0]);
wn_1 = np.imag(l_[2]);
wn_2 print(f"Natural frequencies (rad/s): {wn_1} and {wn_2}")
Natural frequencies (rad/s): 9.38179378603641 and 2.726992997943728
Free response
Let’s compute the free response to some initial conditions. The free state response is given by $$\maybeeq{ \begin{align*} \bm{x}(t) = \Phi(t) \bm{x}(0). \end{align*} }$$
So we can find this from the state transition matrix Φ, which is known from to be .
First, we construct Φ′ symbolically.
"t", real=True)
sp.var(= sp.diag(*list(sp.Matrix(l_)*t)) # Eigenvalue matrix Λ (symbolic)
L = sp.Matrix(M_) # Modal matrix (symbolic)
M = sp.exp(L)
Phi_p pprint(Phi_p)
Matrix([
[1.0*exp(9.38179378603641*I*t), 0, 0, 0],
[ 0, 1.0*exp(-9.38179378603641*I*t), 0, 0],
[ 0, 0, 1.0*exp(2.72699299794373*I*t), 0],
[ 0, 0, 0, 1.0*exp(-2.72699299794373*I*t)]])
Now we can apply our transformation.
= M*Phi_p*M.inv() Phi
So our symbolic solution is to multiply the initial conditions by this matrix.
= sp.Matrix([[1], [0], [0], [0]]) # Initial condition
x_0 = Phi*x_0 # Free response (symbolic, messy) x
Plotting a free response
Let’s make the symbolic solution into something we can evaluate numerically and plot, a Numpy function.
= sp.lambdify(t,x) x_fun
Now let’s set up our time array and state solution for the plot.
= np.linspace(0,5,300)
t_ = np.squeeze(
x_
np.real(x_fun(t_)) )
Plot the state responses through time. The output is shown below.
= plt.subplots()
fig, ax
ax.plot(t_, x_.T)'time (s)')
ax.set_xlabel('state free response')
ax.set_ylabel('$x_1$', '$x_2$', '$x_3$', '$x_4$']) ax.legend([
<matplotlib.legend.Legend at 0x127e64e30>

With a little damping
Now consider the case when the damping coefficent B is nonzero. Let’s recompute A and the eigendecomposition.
= np.array(A.subs(p).subs(pB2), dtype=float)
A_2 print(A_2)
[[-200. -10. 200. 0. ]
[ 8. 0. -8. 0. ]
[ 18.18181818 0.90909091 -18.18181818 -0.90909091]
[ 0. 0. 9. 0. ]]
To explore the modes of vibration, we consider the eigendecomposition of A.
= np.linalg.eig(A_2) l_,M_
Let’s take a closer look at the eigenvalues.
print(l_)
[-2.17777946e+02+0.j -1.53514941e-03+2.73840736j
-1.53514941e-03-2.73840736j -4.00801807e-01+0.j ]
We can see that one of the second-order systems is now “overdamped” or, equivalently, has split into two first-order systems. The other is now underdamped (but barely damped). Let’s compute the natural frequency of the remaining vibratory mode.
= np.imag(l_[1]);
wn_1 print(f"Natural frequency (rad/s): {wn_1}")
Natural frequency (rad/s): 2.7384073593287575
So the effect of damping was to eliminate the ≈ 10 rad/s mode and leave us with a slightly modified version of the ≈ 2.7 rad/s mode.
Free response
Let’s compute the free response to some initial conditions. The free state response is given by $$\maybeeq{ \begin{align*} \bm{x}(t) = \Phi(t) \bm{x}(0). \end{align*} }$$
So we can find this from the state transition matrix Φ, which is known from to be .
First, we construct Φ′ symbolically.
= sp.diag(*list(sp.Matrix(l_)*t)) # Eigenvalue matrix Λ (symbolic)
L = sp.Matrix(M_) # Modal matrix (symbolic)
M = sp.exp(L)
Phi_p pprint(Phi_p)
Matrix([
[1.0*exp(-217.777946076145*t), 0, 0, 0],
[ 0, 1.0*exp(t*(-0.00153514941381959 + 2.73840735932876*I)), 0, 0],
[ 0, 0, 1.0*exp(t*(-0.00153514941381959 - 2.73840735932876*I)), 0],
[ 0, 0, 0, 1.0*exp(-0.400801806845378*t)]])
Now we can apply our transformation.
= M*Phi_p*M.inv() Phi
So our symbolic solution is to multiply the initial conditions by this matrix.
= sp.Matrix([[1], [0], [0], [0]]) # Initial condition
x_0 = Phi*x_0 # Free response (symbolic, messy) x
Plotting a free response
Let’s make the symbolic solution into something we can evaluate numerically and plot, a Numpy function.
= sp.lambdify(t,x) x_fun
Now let’s set up our time array and state solution for the plot.
= np.linspace(0,5,300)
t_ = np.squeeze(
x_
np.real(x_fun(t_)) )
Plot the state responses through time. The output is shown below.
= plt.subplots()
fig, ax
ax.plot(t_, x_.T)'time (s)')
ax.set_xlabel('state free response')
ax.set_ylabel('$x_1$', '$x_2$', '$x_3$', '$x_4$']) ax.legend([
<matplotlib.legend.Legend at 0x137a6acf0>

Online Resources for Section 7.5
No online resources.