Uge 9b: Integration of Vector Fields: The Line Integral and Flux#

Demo by Christian Mikkelstrup, Hans Henrik Hermansen, Jakob Lemvig, Karl Johan Måstrup Kristensen, and Magnus Troen

from sympy import *
from dtumathtools import *

init_printing()

Brief Intro to Vector Fields in SymPy#

In general when working with vector fields in SymPy it is often easiest to defined them as lambda/python functions, since we often are about to insert a vector function into \(\pmb{V}\).

x, y, z = symbols("x,y,z", real=True)

V = lambda x, y, z: Matrix([y * cos(x), y * sin(x), z])
V(x, y, z)
\[\begin{split}\displaystyle \left[\begin{matrix}y \cos{\left(x \right)}\\y \sin{\left(x \right)}\\z\end{matrix}\right]\end{split}\]

If we for instance have an expression of a parameter function

\[\begin{equation*} \boldsymbol{r}_t = \begin{bmatrix} r_1(t) \\ r_2(t) \\ r_3(t) \end{bmatrix} \end{equation*}\]

then we can easily insert \(\boldsymbol{r}_t\) into \(V\) by doing:

r1, r2, r3 = symbols("r1,r2,r3", cls=Function)
t = symbols("t")
r = Matrix([r1(t), r2(t), r3(t)])

# r inserted into V
V(*r)
\[\begin{split}\displaystyle \left[\begin{matrix}r_{2}{\left(t \right)} \cos{\left(r_{1}{\left(t \right)} \right)}\\r_{2}{\left(t \right)} \sin{\left(r_{1}{\left(t \right)} \right)}\\r_{3}{\left(t \right)}\end{matrix}\right]\end{split}\]

If we want to use \(\pmb{V}\), we can analytically simply insert symbolic variables. In later/other courses or in the project period in this course one may get the need for computing vector field properties such as curl and divergence. This is done easily with dtumathtools:

rotV = dtutools.rot(V(x, y, z), (x, y, z))
rotV
\[\begin{split}\displaystyle \left[\begin{matrix}0\\0\\y \cos{\left(x \right)} - \cos{\left(x \right)}\end{matrix}\right]\end{split}\]
divV = dtutools.div(V(x, y, z), (x, y, z))
divV
../_images/7e6f62d1a0212bead1852ac35e128c35e6b2a597dd771447a656195078cb3a8a.png

If one is to carry out sufficiently many analytical computations on \(\pmb{V}\), it may though turn out to be more optimal to just define \(\pmb{V}\) as a SymPy object instead and then use SymPy’s \(\verb|.subs()|\) for inserting variables:

V = Matrix([y * cos(x), y * sin(x), z])
V
\[\begin{split}\displaystyle \left[\begin{matrix}y \cos{\left(x \right)}\\y \sin{\left(x \right)}\\z\end{matrix}\right]\end{split}\]

Line Integral of Vector Fields (Tangential Line Integral)#

We consider the vector field

\[\begin{equation*} \pmb{V}(x,y,z) = \begin{bmatrix} -y \\ x \\ 2z \end{bmatrix} \end{equation*}\]

as well as two curves \(K_1\) and \(K_2\) with the parametric representations

\[\begin{equation*} r_1(u) = \begin{bmatrix} \cos(u) \\ \sin(u) \\ \frac{u}{2} \end{bmatrix} \end{equation*}\]
\[\begin{equation*} r_2(u) = \begin{bmatrix} 1 \\ 0 \\ \frac{u}{2} \end{bmatrix}, \end{equation*}\]

where \(u \in [0,4\pi]\).

x, y, z, u = symbols("x y z u", real=True)
r1 = Matrix([cos(u), sin(u), u / 2])
r2 = Matrix([1, 0, u / 2])
V = Matrix([-y, x, 2 * z])
u_range = (u, 0, 4 * pi)

K1 = dtuplot.plot3d_parametric_line(
    *r1, u_range, show=False, rendering_kw={"color": "red"}, colorbar=False
)
K2 = dtuplot.plot3d_parametric_line(
    *r2, u_range, show=False, rendering_kw={"color": "blue"}, colorbar=False
)
vectorfield_V = dtuplot.plot_vector(
    V,
    (x, -1, 1),
    (y, -1, 1),
    (z, 0, 6),
    n=5,
    quiver_kw={"alpha": 0.5, "length": 0.1, "color": "black"},
    colorbar=False,
    show=False,
)

combined = K1 + K2 + vectorfield_V
combined.legend = False
combined.show()
../_images/6a92e8412632dc03903151e0e43e98b0ab3b0fa79a8d34b6cfd90bd930725e73.png

We want to compute the tangential line integral of \(\pmb{V}\) along each of the two curves from the point \(a = (1,0,0)\) to \(b = (1,0,2\pi)\). This corresponds for both of them to \(u \in [0,4\pi]\). First, we will find the tangent vectors:

r1d = r1.diff(u)
r2d = r2.diff(u)
r1d, r2d
\[\begin{split}\displaystyle \left( \left[\begin{matrix}- \sin{\left(u \right)}\\\cos{\left(u \right)}\\\frac{1}{2}\end{matrix}\right], \ \left[\begin{matrix}0\\0\\\frac{1}{2}\end{matrix}\right]\right)\end{split}\]

The functions that are to be integrated are according to the theorem:

integrand1 = V.subs({x: r1[0], y: r1[1], z: r1[2]}).dot(r1d)
integrand2 = V.subs({x: r2[0], y: r2[1], z: r2[2]}).dot(r2d)
integrand1.simplify(), integrand2.simplify()
../_images/857d13e0810fe2a6910cc3785be84c08c58d18f202fbb81daaa363d597927644.png

The tangential line integral of \(f\) along the curve \(K_1\) is then

\[\begin{equation*} \int_{K_1} \pmb{V} \cdot \mathrm{d}\pmb{s} = \int_0^{4\pi} \frac{u}{2} + 1 \,\mathrm{d}u \end{equation*}\]
integrate(integrand1, (u, 0, 4 * pi))
../_images/4bcca4b541378147f1285c39cee4e1ab31b78ba9117a4463f595cef676580081.png

The tangential line integral of \(f\) along the curve \(K_2\) is then

\[\begin{equation*} \int_{K_2} \pmb{V} \cdot \mathrm{d}\pmb{s} = \int_0^{4\pi} \frac{u}{2} \,\mathrm{d}u \end{equation*}\]
integrate(integrand2, (u, 0, 4 * pi))
../_images/4cf29436186e0d37caac15fdc74e45db04d452c0046563ee588daf4cc241416f.png

Perspective:#

If we interpret \(\pmb{V}\) as a force vector field, then the work (energy) with which \(\pmb{V}\) constributes when the particles is moved along the vertical line is \(4\pi\) less than if the particle was moved along the spiral curve. The magnitude of the work is hence in this example dependent on the path. \(\pmb{V}\) can hence not be a gradient vector field.

Seaching for Anti-Derivative via Stair Lines. Theory#

An anti-derivative to a smooth vector field \(\pmb{V}\) can, if an anti-derivative exists, be determined via the tangential line integral of \(\pmb{V}\) along an arbitrary curve from the origin \(\pmb{x}_0 = (0,0,0)\) to an arbitrary fixed point \(\pmb{x} = (x, y, z)\). One achives the easiest computations by using the stair line from the origin to \(P\) as the curve.

u_range = (u, 0, 1)
r1 = dtuplot.plot3d_parametric_line(
    u, 0, 0, u_range, show=False, rendering_kw={"color": "red"}, colorbar=False
)
r2 = dtuplot.plot3d_parametric_line(
    1, u, 0, u_range, show=False, rendering_kw={"color": "red"}, colorbar=False
)
r3 = dtuplot.plot3d_parametric_line(
    1, 1, u, u_range, show=False, rendering_kw={"color": "red"}, colorbar=False
)
xyz = dtuplot.scatter(Matrix([1, 1, 1]), show=False, rendering_kw={"color": "black"})
combined = r1 + r2 + r3 + xyz
combined.legend = False
combined.camera = {"azim": 37, "elev": 16}

combined.show()
../_images/33ba8fa95895e25e20138393e13de5ec46dbebf46535188f2bc4021787097258.png

The plot above shows the three stair lines which the methods utilizes. We will start with the following parametrization of the lines:

r1 = Matrix([u, 0, 0])
r2 = Matrix([x, u, 0])
r3 = Matrix([x, y, u])

Here we have, respectively, \(u \in [0,x]\), \(u \in [0,y]\), and \(u \in [0,z]\). An advantage with this method is that the three tangent vectors are very simple.

r1d = r1.diff(u)
r2d = r2.diff(u)
r3d = r3.diff(u)
r1d, r2d, r3d
\[\begin{split}\displaystyle \left( \left[\begin{matrix}1\\0\\0\end{matrix}\right], \ \left[\begin{matrix}0\\1\\0\end{matrix}\right], \ \left[\begin{matrix}0\\0\\1\end{matrix}\right]\right)\end{split}\]

Now consider an arbitrary smoot vector field

\[\begin{equation*} \pmb{V}(x,y,z) = \begin{bmatrix} V_1(x,y,z) \\ V_2(x,y,z) \\ V_3(x,y,z) \end{bmatrix} \end{equation*}\]

The tangential line integral of \(\pmb{V}\) along the stair line consists of the sum of the tangential line integral of \(\pmb{V}\) along each of the three line segments of which the stair consists.
We consider the three integrands that are involved in the three integrals:

\[\begin{equation*} \pmb{V}(r_1(u)) \cdot r_1'(u) = V_1(u,0,0) \end{equation*}\]
\[\begin{equation*} \pmb{V}(r_2(u)) \cdot r_2'(u) = V_1(x,u,0) \end{equation*}\]
\[\begin{equation*} \pmb{V}(r_3(u)) \cdot r_3'(u) = V_1(x,y,u) \end{equation*}\]

Hence we have the following formula for the tangential line integral of \(\pmb{V}\) along a stair line:

\[\begin{equation*} \int_T \pmb{V} \cdot \mathrm{d}\pmb{s} = \int_0^x V_1(u,0,0) \,\mathrm{d}u + \int_0^y V_2(x,u,0) \,\mathrm{d}u +\int_0^z V_3(x,y,u) \,\mathrm{d}u \end{equation*}\]

Searching for Anti-Derivative via Stair Lines. Example#

We are to investigate whether the following vector field \(\pmb{V}\) is a gradient vector field:

\[\begin{equation*} \pmb{V}(x,y,z) = \begin{bmatrix} y^2 + z \\ 2yz^2 + 2yx \\ 2y^2z + x \end{bmatrix}. \end{equation*}\]

We compute the tangential line integral of \(\pmb{V}\) along the stair line by use of the formula:

\[\begin{equation*} \int_T \pmb{V} \cdot \mathrm{d}\pmb{s} = \int_0^x V_1(u,0,0) \,\mathrm{d}u + \int_0^y V_2(x,u,0) \,\mathrm{d}u +\int_0^z V_3(x,y,u) \,\mathrm{d}u. \end{equation*}\]

We find the three integrands:

V = Matrix([y**2 + z, 2 * y * z**2 + 2 * y * x, 2 * y**2 * z + x])
integrand1 = V[0].subs({x: u, y: 0, z: 0})
integrand2 = V[1].subs({y: u, z: 0})
integrand3 = V[2].subs({z: u})

integrand1, integrand2, integrand3
../_images/290fca0d7f736c0e378c7d8f557eac4e6cc75ba94af04463b4726b722ac15e02.png

The tangential line integral of \(\pmb{V}\) along the stair line is then:

F = (
    integrate(integrand1, (u, 0, x))
    + integrate(integrand2, (u, 0, y))
    + integrate(integrand3, (u, 0, z))
)
F
../_images/0aff10dc8f3aff3bc3ff3bec06b3c258f783d5c8d0e6a2d4d0f1b4d084920eb2.png

If we consider the line integral as a function \(F\) of \((x,y,z)\), we can now test whether \(F\) really is an anti-derivative to \(\pmb{V}\).

V_F = dtutools.gradient(F)
V_F, Eq(V_F, V)
\[\begin{split}\displaystyle \left( \left[\begin{matrix}y^{2} + z\\2 x y + 2 y z^{2}\\x + 2 y^{2} z\end{matrix}\right], \ \text{True}\right)\end{split}\]

Since \(\pmb{V}\) is identical to the gradient of \(F\), \(F\) is an anti-derivative to \(\pmb{V}\)! This means that the tangential line integral of \(\pmb{V}\) along an arbitrary closed curve always will result in \(0\). Let us check this claim on the following knot:

t = symbols("t")
knot = (
    Matrix(
        [
            -10 * cos(t) - 2 * cos(5 * t) + 15 * sin(2 * t),
            -15 * cos(2 * t) + 10 * sin(t) - 2 * sin(5 * t),
            10 * cos(3 * t),
        ]
    )
    * S(1)
    / 10
)
knot
\[\begin{split}\displaystyle \left[\begin{matrix}\frac{3 \sin{\left(2 t \right)}}{2} - \cos{\left(t \right)} - \frac{\cos{\left(5 t \right)}}{5}\\\sin{\left(t \right)} - \frac{\sin{\left(5 t \right)}}{5} - \frac{3 \cos{\left(2 t \right)}}{2}\\\cos{\left(3 t \right)}\end{matrix}\right]\end{split}\]
dtuplot.plot3d_parametric_line(
    *knot, (t, -pi, pi), rendering_kw={"color": "blue"}, legend=False
)
../_images/28df684d7b889535ad3cb82aa85f9f3328e3da54fa2e51be84939db00cf3df1a.png
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7fbc729cacd0>
## Note -- This integral takes a long time for SymPy to compute. It might easily take more than a minute, so be patient
#integrate(V.subs({x: knot[0], y: knot[1], z: knot[2]}).dot(knot.diff(t)), (t, -pi, pi))

The work which the force field \(\pmb{V}\) contributes with as the particle moves a trip around the know is thus \(0\). Just as expected, since \(\pmb{V}\) is a gradient vector field.

Surface Integral of Vector Fields (Flux through Surfaces)#

We are given a vector field by:

x, y, z = symbols("x y z", real=True)
V = lambda x, y, z: Matrix([x, y, 2])
V(x, y, z)
\[\begin{split}\displaystyle \left[\begin{matrix}x\\y\\2\end{matrix}\right]\end{split}\]

We consider the part of the standard paraboloid \(F_{parab}\) that is located below the plane \(z=1\). This can be parametrized by:

u, v = symbols("u v", real=True)
r = Matrix([u * cos(v), u * sin(v), u**2])
r
\[\begin{split}\displaystyle \left[\begin{matrix}u \cos{\left(v \right)}\\u \sin{\left(v \right)}\\u^{2}\end{matrix}\right]\end{split}\]

where \(u\in [0,1]\) and \(v\in [-\pi,\pi]\). The vector field and the surface are illustrated:

p_parab = dtuplot.plot3d_parametric_surface(
    *r,
    (u, 0, 1),
    (v, -pi, pi),
    use_cm=False,
    camera={"elev": 25, "azim": -55},
    show=False
)
p_field = dtuplot.plot_vector(
    V(x, y, z),
    (x, -1, 1),
    (y, -1, 1),
    (z, 0, 1.2),
    n=4,
    use_cm=False,
    quiver_kw={"alpha": 0.5, "length": 0.1, "color": "red"},
    show=False,
)

(p_parab + p_field).show()
../_images/5a05960696cbaf905827eddf6d8f8d979d29e959ab4144398be3024ea3b168c7.png

We can do with just plotting the vector field on the surface itself:

# Vectors arrive from all corners. There are too many
# vectors when "p_parab" is used, so here we reduce n
dummy_surface = dtuplot.plot3d_parametric_surface(
    *r, (u, 0, 1), (v, -pi, pi), n=8, show=False
)
p_field2 = dtuplot.plot_vector(
    V(x, y, z),
    (x, -1, 1),
    (y, -1, 1),
    (z, 0, 1),
    slice=dummy_surface[0],
    use_cm=False,
    quiver_kw={"alpha": 0.5, "length": 0.1, "color": "red"},
    show=False,
)
# Better camera angle for looking at the vectors with
p_parab.camera = {"elev": 60, "azim": -50}

(p_parab + p_field2).show()
../_images/482a8bc6c7db3828e10c77d6a136dd2d2c5aadfc803d8baffa6fa8ffff78a558.png

Computation of Flux “down” through the Paraboloid \(F_{parab}\)#

This clearly looks as though there is a (positive) flux up through the surface. We would thus expect to see that the flux down through the surface is negative. We will start by finding the normal vector to the parabloid given by our chosen parametrization:

n_parab = r.diff(u).cross(r.diff(v))
simplify(n_parab)
\[\begin{split}\displaystyle \left[\begin{matrix}- 2 u^{2} \cos{\left(v \right)}\\- 2 u^{2} \sin{\left(v \right)}\\u\end{matrix}\right]\end{split}\]

Notice here that the normal vector \(n\) is pointing in the “wrong” direction (inwards towards the \(z\) axis and thus with a postive \(z\) coordinate); “wrong” in the sense that we actually wish to find the flux down through the surface. We thus simply have to remember to change the sign of the normal vector \(n_{parab}\):

n_parab = -simplify(n_parab)
n_parab
\[\begin{split}\displaystyle \left[\begin{matrix}2 u^{2} \cos{\left(v \right)}\\2 u^{2} \sin{\left(v \right)}\\- u\end{matrix}\right]\end{split}\]

The length of the normal vector is only needed for arguing that the parametrization is regular:

n_parab.norm().simplify()
../_images/54a1b5b3b79f71d59f208d64b4466b33d3d2c1d519ba1a60002d4b928a8c27ad.png

This expression is positive on the interior of \(\Gamma\), so it is indeed regular. We will now find the integrand:

integrand = n_parab.dot(V(*r))
simplify(integrand)
../_images/f3d18c595fd91ea7a8fbd260e6ae953031fe68487cdd2c2a6bb162d3ada691d4.png

with which the flux is computed by doing

(4)#\[\begin{equation} \mathrm{Flux}(\pmb{V}, F_{parab}) = \int_0^1\int_{-\pi}^\pi \pmb{V}(r(u,v))\cdot n_{parab}\,\mathrm{d}u\,\mathrm{d}v \end{equation}\]
integrate(integrand, (u, 0, 1), (v, -pi, pi))
../_images/35f5cc0e42b30c11a4e85d1edecc4b8b7a0962843792ce07e2ab8d534406cb2f.png

The wanted flux is thus \(-\pi\)!

Computation of the Flux through the “Lid” \(F_{lid}\) of the Paraboloid in the (Positive) Direction of the \(z\) axis#

r2 = Matrix([u * cos(v), u * sin(v), 1])
n2 = r2.diff(u).cross(r2.diff(v))
simplify(n2)
\[\begin{split}\displaystyle \left[\begin{matrix}0\\0\\u\end{matrix}\right]\end{split}\]

The direction of the normal vector fits with the wanted direction through the surface \(F_{lid}\).

integrand2 = n2.dot(V(*r2))
integrate(integrand2, (u, 0, 1), (v, -pi, pi))
../_images/46c7d195b5b0b7a95cd9748c4760d952374bc08b95fe0c21e993f68d5eedb159.png

The flux through the lid is \(\mathrm{Flux}(\pmb{V},F_{lid}) = 2\pi\).

The Flux through the closed Surface with Paraboloid and Lid \(F_{closed} = F_{parab} \cup F_{lid}\)#

p_lid = dtuplot.plot3d_parametric_surface(
    *r2,
    (u, 0, 1),
    (v, -pi, pi),
    use_cm=False,
    n1=4,
    n2=16,
    camera={"elev": 25, "azim": -55},
    show=False
)
p_fieldlid = dtuplot.plot_vector(
    V(x, y, z),
    (x, -1, 1),
    (y, -1, 1),
    (z, 0, 1),
    slice=p_lid[0],
    use_cm=False,
    quiver_kw={"alpha": 0.5, "length": 0.1, "color": "red"},
    show=False,
)
p_parab.camera = {"elev": 15, "azim": -60}

(p_parab + p_field2 + p_lid + p_fieldlid).show()
../_images/845b8b7e8a8d0fda12010d8b5cd30b5dfbd30f1232cc5e2ecc10e7fc749a0ef3.png

We do not need to continue with any further calculations! We here have the flux through all surfaces, from which the total flux through the closed surface is \(Flux(\pmb{V},F_{closed})=-\pi+2\pi=\pi\).

Flux through a Piece of a Spherical Surface#

We are given a vector field,

x, y, z = symbols("x y z")
V = Matrix([-y + x, x, z])
V
\[\begin{split}\displaystyle \left[\begin{matrix}x - y\\x\\z\end{matrix}\right]\end{split}\]

and a section of a spherical surface,

u, v, w = symbols("u v w", real=True)
radius = 2
r = radius * Matrix([sin(u) * cos(v), sin(u) * sin(v), cos(u)])
r
\[\begin{split}\displaystyle \left[\begin{matrix}2 \sin{\left(u \right)} \cos{\left(v \right)}\\2 \sin{\left(u \right)} \sin{\left(v \right)}\\2 \cos{\left(u \right)}\end{matrix}\right]\end{split}\]

with \(u\in[\pi/6,\pi/2]\) and \(v\in[0,\pi]\),

p_surface = dtuplot.plot3d_spherical(
    radius,
    (u, pi / 6, pi / 2),
    (v, 0, pi),
    aspect="equal",
    camera={"elev": 25, "azim": 55},
    show=False,
)
dummy_surface = dtuplot.plot3d_spherical(
    radius, (u, pi / 6, pi / 2), (v, 0, pi), show=False, n=8
)
p_field = dtuplot.plot_vector(
    V,
    (x, -1, 1),
    (y, -1, 1),
    (z, 0, 1),
    slice=dummy_surface[0],
    use_cm=False,
    quiver_kw={"alpha": 0.5, "length": 0.2, "color": "red"},
    show=False,
)

(p_surface + p_field).show()
../_images/8f9434fa89270f774e8566990f4f8fe68289994c405abd02f435c39a8a6e49e4.png

The flux is computed to be

n = r.diff(u).cross(r.diff(v))
integrand = n.dot(V.subs({x: r[0], y: r[1], z: r[2]}))
integrate(integrand, (u, pi / 6, pi / 2), (v, 0, pi))
../_images/3e59e66f525f40b3c120787df2476175b41fc71fca56eb81de862df4e339fd7d.png

Flux through a Section of a spherical surface using Gauss’ Divergence Theorem (not Syllabus)#

We will compare the result from the flux computation with a result from using Gauss’ Theorem through a semi-sphere with a radius \(a\) (\(u\in[0,a]\), \(v\in[0,\pi/2]\), and \(w\in[-\pi,\pi]\)), with the flux through its two surfaces. This should give the same result!

V = Matrix([8 * x, 8, 4 * z**3])
r = u * Matrix([sin(v) * cos(w), sin(v) * sin(w), cos(v)])
V, r
\[\begin{split}\displaystyle \left( \left[\begin{matrix}8 x\\8\\4 z^{3}\end{matrix}\right], \ \left[\begin{matrix}u \sin{\left(v \right)} \cos{\left(w \right)}\\u \sin{\left(v \right)} \sin{\left(w \right)}\\u \cos{\left(v \right)}\end{matrix}\right]\right)\end{split}\]

First, using Gauss’ Theorem,

a = symbols("a", real=True, positive=True)
M = Matrix.hstack(r.diff(u), r.diff(v), r.diff(w))
# the determinatn is positive. The only term that is not
# squared is sin(v), which is positive within v's limits
jacobian = M.det()
divV = dtutools.div(V, [x, y, z])
divV_r = divV.subs({x: r[0], y: r[1], z: r[2]})
integrate(divV_r * jacobian, (u, 0, a), (v, 0, pi / 2), (w, -pi, pi))
../_images/69683868b30b9e55b273a0231d623d79845dc99625877b38adc2fd9c776e3574.png

and then through the two surfaces,

# The first surface corresponds to fixing u as the radius
r1 = r.subs(u, a)
n1 = r1.diff(v).cross(r1.diff(w))  # we have checked that it points outwards
integrand1 = n1.dot(V.subs({x: r1[0], y: r1[1], z: r1[2]}))
flux1 = integrate(integrand1, (v, 0, pi / 2), (w, -pi, pi))
# Second surface corresponds to fixing v to pi/2
# The remaining becomes the disc with radius a
r2 = r.subs(v, pi / 2)
n2 = -r2.diff(u).cross(r2.diff(w))  # must be negative in order to point outwards
integrand2 = n2.dot(V.subs({x: r2[0], y: r2[1], z: r2[2]}))
flux2 = integrate(integrand2, (u, 0, a), (w, -pi, pi))
flux1 + flux2
../_images/69683868b30b9e55b273a0231d623d79845dc99625877b38adc2fd9c776e3574.png

As expected!

Flow Curves / Integral Curves#

Vector fields can also “determine” a particle’s motion in the plane or in 3D space (a flow curve). We image that we at time \(t=0\) throw a particle into a force field at the point \(\pmb{x}_0\), and we wish to model the particle’s trajectory (curve) \(\pmb{r}(t)\) as a function of time. These curves are called integral curves (and sometimes flow curves). They are solutions to the differential equation system:

\[\begin{equation*} \pmb{r}'(t) = \pmb V(\pmb{r}(t)), \quad \pmb{r}(0) = \pmb{x}_0 \end{equation*}\]

where \(\pmb{x}_0\) is the initial point.

We consider the vector field

\[\begin{equation*} \pmb{V}(x,y) = \left[\begin{matrix}-\frac{1}{4}x + \frac{1}{2}y\\\frac{1}{2}x + \frac{1}{4}y\end{matrix}\right], \end{equation*}\]

and two points that indicate the starting points \(s_1\) and \(s_2\) of two particles within a vector field \(V\)

\[\begin{equation*} s_1 = (5, 0),\quad s_2 = (-3, \frac{1}{2}). \end{equation*}\]
s1, s2 = Matrix([5, 0]), Matrix([-3, S(1)/2])

Can we now find the flow curves of these two particles? We denote them, respectively, \(r_1\) and \(r_2\).

In this case the differential equation system is linear, so we know from the solution model from Mathematics 1a. We will first find the system matrix:

\[\begin{equation*} A = \left[\begin{matrix}-\frac{1}{4} & \frac{1}{2}\\\frac{1}{2} & \frac{1}{4}\end{matrix}\right] \end{equation*}\]

and we can use the eigenvalues and corresponding eigenvectors of \(A\) to determine the flow curves

A = Matrix(2, 2, [S("-1/4"), S("1/2"), S("1/2"), S("1/4")])
A.eigenvects(), A.diagonalize()
\[\begin{split}\displaystyle \left( \left[ \left( - \frac{\sqrt{5}}{4}, \ 1, \ \left[ \left[\begin{matrix}- \frac{\sqrt{5}}{2} - \frac{1}{2}\\1\end{matrix}\right]\right]\right), \ \left( \frac{\sqrt{5}}{4}, \ 1, \ \left[ \left[\begin{matrix}- \frac{1}{2} + \frac{\sqrt{5}}{2}\\1\end{matrix}\right]\right]\right)\right], \ \left( \left[\begin{matrix}- \frac{\sqrt{5}}{2} - \frac{1}{2} & - \frac{1}{2} + \frac{\sqrt{5}}{2}\\1 & 1\end{matrix}\right], \ \left[\begin{matrix}- \frac{\sqrt{5}}{4} & 0\\0 & \frac{\sqrt{5}}{4}\end{matrix}\right]\right)\right)\end{split}\]

Both \(\verb|A.eigenvects()|\) and \(\verb|A.diagonalize()|\) provide the wanted info, and which one uses is just a question of preference.

With \(\verb|A.eigenvects()|\):#

k = symbols("k1:3")
ev = A.eigenvects()
ev
\[\begin{split}\displaystyle \left[ \left( - \frac{\sqrt{5}}{4}, \ 1, \ \left[ \left[\begin{matrix}- \frac{\sqrt{5}}{2} - \frac{1}{2}\\1\end{matrix}\right]\right]\right), \ \left( \frac{\sqrt{5}}{4}, \ 1, \ \left[ \left[\begin{matrix}- \frac{1}{2} + \frac{\sqrt{5}}{2}\\1\end{matrix}\right]\right]\right)\right]\end{split}\]

The general solution is set up as we know it from system of first-order differential equations (with two different eigenvalues):

\[\begin{equation*} \boldsymbol r(u) = k_1 e^{{\lambda_1}u} \boldsymbol v_1 + k_2 e^{{\lambda_2}u} \boldsymbol v_2 \end{equation*}\]
r_general = (
    k[0] * exp(ev[0][0] * u) * ev[0][-1][0] + k[1] * exp(ev[1][0] * u) * ev[1][-1][0]
)
r_general
\[\begin{split}\displaystyle \left[\begin{matrix}k_{1} \left(- \frac{\sqrt{5}}{2} - \frac{1}{2}\right) e^{- \frac{\sqrt{5} u}{4}} + k_{2} \left(- \frac{1}{2} + \frac{\sqrt{5}}{2}\right) e^{\frac{\sqrt{5} u}{4}}\\k_{1} e^{- \frac{\sqrt{5} u}{4}} + k_{2} e^{\frac{\sqrt{5} u}{4}}\end{matrix}\right]\end{split}\]

The given starting points are our initial conditions, and the equations

\[\begin{equation*} \boldsymbol r_1(0) = s_1\quad \text{and}\quad \boldsymbol r_1(0) = s_2 \end{equation*}\]

are solved in order to find our specific flow curves

ks_r1 = solve(r_general.subs(u, 0) - s1)
ks_r2 = solve(r_general.subs(u, 0) - s2)
ks_r1, ks_r2
../_images/600adb32cde75868873f340a360e62f91f99d24ee61a59e3cb5991cf6d9bc06e.png

These can be substituted directly into \(\boldsymbol r(u)\)

r1 = r_general.subs(ks_r1)
r2 = r_general.subs(ks_r2)

and we arrive at the flow curves

r1, r2
\[\begin{split}\displaystyle \left( \left[\begin{matrix}\sqrt{5} \left(- \frac{1}{2} + \frac{\sqrt{5}}{2}\right) e^{\frac{\sqrt{5} u}{4}} - \sqrt{5} \left(- \frac{\sqrt{5}}{2} - \frac{1}{2}\right) e^{- \frac{\sqrt{5} u}{4}}\\\sqrt{5} e^{\frac{\sqrt{5} u}{4}} - \sqrt{5} e^{- \frac{\sqrt{5} u}{4}}\end{matrix}\right], \ \left[\begin{matrix}\left(- \frac{1}{2} + \frac{\sqrt{5}}{2}\right) \left(\frac{1}{4} - \frac{11 \sqrt{5}}{20}\right) e^{\frac{\sqrt{5} u}{4}} + \left(\frac{1}{4} + \frac{11 \sqrt{5}}{20}\right) \left(- \frac{\sqrt{5}}{2} - \frac{1}{2}\right) e^{- \frac{\sqrt{5} u}{4}}\\\left(\frac{1}{4} - \frac{11 \sqrt{5}}{20}\right) e^{\frac{\sqrt{5} u}{4}} + \left(\frac{1}{4} + \frac{11 \sqrt{5}}{20}\right) e^{- \frac{\sqrt{5} u}{4}}\end{matrix}\right]\right)\end{split}\]
v_field = dtuplot.plot_vector(
    A * Matrix([x, y]),
    n=10,
    scalar=False,
    colorbar=False,
    quiver_kw={"color": "black"},
    show=False,
)
r1_plot = dtuplot.plot_parametric(
    *r1, [u, 0, 3], rendering_kw={"color": "red"}, colorbar=False, show=False
)
r2_plot = dtuplot.plot_parametric(
    *r2, (u, 0, 4), rendering_kw={"color": "blue"}, colorbar=False, show=False
)
(v_field + r1_plot + r2_plot).show()
No ranges were provided. This function will attempt to find them, however the order will be arbitrary, which means the visualization might be flipped.
../_images/424b27939ba05dc91fc441fcdf18f7915a0f8bd2581b2653eb1f16339fb24d74.png

Solution via \(\verb|A.diagonalize()|\):#

k = symbols("k1:3")
matrix_of_evs, LAMBDA = A.diagonalize()
matrix_of_evs, LAMBDA
\[\begin{split}\displaystyle \left( \left[\begin{matrix}- \frac{\sqrt{5}}{2} - \frac{1}{2} & - \frac{1}{2} + \frac{\sqrt{5}}{2}\\1 & 1\end{matrix}\right], \ \left[\begin{matrix}- \frac{\sqrt{5}}{4} & 0\\0 & \frac{\sqrt{5}}{4}\end{matrix}\right]\right)\end{split}\]
r_general = k[0] * exp(LAMBDA[0, 0] * u) * matrix_of_evs.col(0) + k[1] * exp(
    LAMBDA[1, 1] * u
) * matrix_of_evs.col(1)
r_general
\[\begin{split}\displaystyle \left[\begin{matrix}k_{1} \left(- \frac{\sqrt{5}}{2} - \frac{1}{2}\right) e^{- \frac{\sqrt{5} u}{4}} + k_{2} \left(- \frac{1}{2} + \frac{\sqrt{5}}{2}\right) e^{\frac{\sqrt{5} u}{4}}\\k_{1} e^{- \frac{\sqrt{5} u}{4}} + k_{2} e^{\frac{\sqrt{5} u}{4}}\end{matrix}\right]\end{split}\]
eq_k_r1 = Eq
ks_r1 = solve(r_general.subs(u, 0) - s1)
ks_r2 = solve(r_general.subs(u, 0) - s2)
ks_r1, ks_r2
../_images/600adb32cde75868873f340a360e62f91f99d24ee61a59e3cb5991cf6d9bc06e.png

and we have the same flow curves as above

r1 = r_general.subs(ks_r1)
r2 = r_general.subs(ks_r2)
r1, r2
\[\begin{split}\displaystyle \left( \left[\begin{matrix}\sqrt{5} \left(- \frac{1}{2} + \frac{\sqrt{5}}{2}\right) e^{\frac{\sqrt{5} u}{4}} - \sqrt{5} \left(- \frac{\sqrt{5}}{2} - \frac{1}{2}\right) e^{- \frac{\sqrt{5} u}{4}}\\\sqrt{5} e^{\frac{\sqrt{5} u}{4}} - \sqrt{5} e^{- \frac{\sqrt{5} u}{4}}\end{matrix}\right], \ \left[\begin{matrix}\left(- \frac{1}{2} + \frac{\sqrt{5}}{2}\right) \left(\frac{1}{4} - \frac{11 \sqrt{5}}{20}\right) e^{\frac{\sqrt{5} u}{4}} + \left(\frac{1}{4} + \frac{11 \sqrt{5}}{20}\right) \left(- \frac{\sqrt{5}}{2} - \frac{1}{2}\right) e^{- \frac{\sqrt{5} u}{4}}\\\left(\frac{1}{4} - \frac{11 \sqrt{5}}{20}\right) e^{\frac{\sqrt{5} u}{4}} + \left(\frac{1}{4} + \frac{11 \sqrt{5}}{20}\right) e^{- \frac{\sqrt{5} u}{4}}\end{matrix}\right]\right)\end{split}\]

An Example with the Gauss Theorem (not Syllabus)#

We are given the vector field

V = Matrix([-x + cos(z), -y * x, 3 * z + exp(y)])
V
\[\begin{split}\displaystyle \left[\begin{matrix}- x + \cos{\left(z \right)}\\- x y\\3 z + e^{y}\end{matrix}\right]\end{split}\]

as well as the spatial set \(\Omega\) given by

\[\begin{equation*} \bigl\{(x,y,z) \in \mathbb{R}^3 : x\in[0,3], \, y\in[0,2], \,z\in[0,y^2] \bigr\} \end{equation*} \]

Here the set is illustrated:

a = symbols("a")
# Since one cannot plot a 3D volumen (not easily, that is), we will plot its surfaces,
p = dtuplot.plot3d_parametric_surface(
    x,
    y,
    y**2,
    (x, 0, 3),
    (y, 0, 2),
    {"color": "royalblue"},
    use_cm=False,
    aspect="equal",
    show=False,
)
p.extend(
    dtuplot.plot3d_parametric_surface(
        0,
        y,
        a * y**2,
        (a, 0, 1),
        (y, 0, 2),
        {"color": "royalblue", "alpha": 0.5},
        use_cm=False,
        aspect="equal",
        show=False,
    )
)
p.extend(
    dtuplot.plot3d_parametric_surface(
        3,
        y,
        a * y**2,
        (a, 0, 1),
        (y, 0, 2),
        {"color": "royalblue", "alpha": 0.5},
        use_cm=False,
        aspect="equal",
        show=False,
    )
)
p.extend(
    dtuplot.plot3d_parametric_surface(
        x,
        2,
        a * 4,
        (x, 0, 3),
        (a, 0, 1),
        {"color": "royalblue", "alpha": 0.5},
        use_cm=False,
        aspect="equal",
        show=False,
    )
)
p.extend(
    dtuplot.plot3d_parametric_surface(
        x,
        y,
        0,
        (x, 0, 3),
        (y, 0, 2),
        {"color": "royalblue", "alpha": 0.5},
        use_cm=False,
        aspect="equal",
        show=False,
    )
)
p_field = dtuplot.plot_vector(
    V,
    (x, 0, 3),
    (y, 0, 2),
    (z, 0, 4),
    n=4,
    use_cm=False,
    quiver_kw={"alpha": 0.5, "length": 0.05, "color": "red"},
    show=False,
)

(p + p_field).show()

We will now determine the flux of the vector field \(\pmb{V}\) out through the closed surface of \(\Omega\) by use of Gauss’ Theorem!

We will first determine a parametric representation of the solid volume,

u, v, w = symbols("u v w")
r = Matrix([u, v, w * v**2])
r
M = Matrix.hstack(r.diff(u), r.diff(v), r.diff(w))
M, det(M)

We can compute the determinant since

\[\begin{equation*} (r_u'(u,v,w)\times r_v'(u,v,w))\cdot r_w'(u,v,w)=\det\begin{bmatrix}|&|&|\\r_u'(u,v,w)&r_v'(u,v,w)&r_w'(u,v,w)\\|&|&|\end{bmatrix}. \end{equation*}\]

We must, though, not forget the absolute value (see Definition 25.19). It might, though, often be possible to achieve a nicer-looking expression by not computing the absolute value if one can garantee that the determinant is always positive. That is the case here, since \(v>0\), so

jacobian = M.det()
jacobian

The divergence now becomes

divV = dtutools.div(V, (x, y, z))
divV

The divergence boudned to the parametrized region is thus found to be,

divV_r = divV.subs({x: r[0], y: r[1], z: r[2]})
divV_r

Since

integrate(divV_r * jacobian, (u, 0, 3), (v, 0, 2), (w, 0, 1))

we get from Gauss’ Theorem that:

(5)#\[\begin{equation} \int_{\delta\Omega}\pmb{V}\cdot \mathrm{d}\pmb{S} = \int_0^1\int_0^2\int_0^3(2-u)\cdot v^2\,\mathrm{d}u\,\mathrm{d}v\,\mathrm{d}w = 4 \end{equation}\]

In this example it was particularly obvious to use Gauss’ Theorem, both since we would avoid computing the flux through \(5\) surfaces, after which we would add up those partial results, and since the divergence is mathematically simple in comparison to its vector field. This must, though, be considered every time a new exercise is given!