Uge 9b: Integration af vektorfelter: Kurveintegralet og Flux#

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

from sympy import *
from dtumathtools import *

init_printing()

Kort intro til vektorfelter i SymPy#

Generelt når vi arbjede med vektorfelter i SymPy er det oftest nemmest at definere dem som lambda-/pythonfunktioner, da vi ofte skal indsætte en vektorfunktion i \(\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}\]

Hvis vi fx har et udtryk for en parameterfunktion

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

kan vi nemt indsætte \(\boldsymbol{r}_t\) i \(V\) ved:

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

# r indsat i 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}\]

Hvis vi vil anvende \(\pmb{V}\), analytisk kan vi bare indsætte symbolske variable. I senere/andre kurser eller i projektperioden i dette kursus, kan man få brug for at udregne fx rotationen og divergensen af et vektorfelt. Dette gøres nemt med 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

Hvis man skal lave tilpas mange analytiske beregninger på \(\pmb{V}\), kan det dog vise sig at være mest optimalt bare at definere \(\pmb{V}\) som et SymPy objekt i stedet, og bruge SymPy’s \(\verb|.subs()|\) til indsætte variable:

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}\]

Kurveintegral af vektorfelter (tangentielt kurveintegral)#

Vi betragter vektorfeltet

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

samt to kurver \(K_1\) og \(K_2\) med parameterfremstillingerne

\[\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*}\]

hvor \(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
)
vektorfelt_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 + vektorfelt_V
combined.legend = False
combined.show()
../_images/6a92e8412632dc03903151e0e43e98b0ab3b0fa79a8d34b6cfd90bd930725e73.png

Vi vil udrenge det tangentielle kurveintegral af \(\pmb{V}\) langs hver af de to kurver fra punktet \(a = (1,0,0)\) til \(b = (1,0,2\pi)\). Dette svarer for begge til \(u \in [0,4\pi]\). Først finder vi tangentvektorene:

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}\]

Funktionerne der skal integreres er i følge sætningen:

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

Det tangentielle integral af \(f\) langs kurven \(K_1\) er da

\[\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

Det tangentielle integral af \(f\) langs kurven \(K_2\) er da

\[\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

Perspektivering:#

Hvis vi opfatter \(\pmb{V}\) som et kraft-vektorfelt , er det arbejde \(\pmb{V}\) bidrager med når partiklen flyttes langs den lodrette linje, \(4\pi\) mindre end når partiklen flyttes langs spiralkurven. Arbejdets størrelse er altså i dette eksempel afhængig af transportvejen. \(\pmb{V}\) kan derfor ikke være et gradient-vektorfelt.

Søgning efter stamfunktion via trappelinjer. Teori#

En stamfunktion til et glat vektorfelt \(\pmb{V}\) kan, hvis en stamfunktion findes, bestemmes ved det tangentielle kurveintegral af \(\pmb{V}\) langs en vilkårlig kurve fra origo \(\pmb{x}_0 = (0,0,0)\) til et vilkårligt fast punkt \(\pmb{x} = (x, y, z)\). De nemmeste udregninger får man ved som kurve at benytte trappelinjen fra Origo til \(P\).

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

Vist på plottet ovenfor er de tre trappelinjer, som metoden består af. Vi vil benytte følgende parametrisering af linjerne:

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

Her har vi hhv. \(u \in [0,x]\), \(u \in [0,y]\) og \(u \in [0,z]\). En fordel ved denne metode er, at de tre tangentvektorer er meget 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}\]

Betragt nu et vilkårligt glat vektorfelt

\[\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*}\]

Det tangentielle kurveintegral af \(\pmb{V}\) langs trappelinjen består af summen af det tangentielle kurveintegral af \(\pmb{V}\) langs hver af de tre linjestykker som trappelinjen består af.
Vi betragter de tre integrander, der indgår i tre integraler:

\[\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*}\]

Vi har derfor følgende formel for det tangentielle kurveintegral af \(\pmb{V}\) langs trappelinjen

\[\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*}\]

Søgning efter stamfunktion via trappelinjer. Eksempel#

Vi skal undersøge om det følgende vektorfelt \(\pmb{V}\) er et gradientvektorfelt

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

Vi bestemmer det tangentielle kurveintegral af \(\pmb{V}\) langs trappelinjen ved hjælp af formlen:

\[\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*}\]

Vi finder de tre integrander:

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

Nu kan vi finde det tangentielle kurveintegral af \(\pmb{V}\) langs trappelinjen er dermed:

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

Hvis vi anser kurveintegralet som en funktion \(F\) af \((x,y,z)\), kan vi nu teste om \(F\) virkelig er en stamfunktion til \(\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}\]

Da \(\pmb{V}\) er identisk med gradienten af \(F\), er \(F\) en stamfunktion til \(\pmb{V}\)! Det betyder at det tangentielle kurveintegral af \(\pmb{V}\) langs en vilkårlig lukket kurve skal give 0. Lad os tjekke det vi på følgende knude:

t = symbols("t")
knude = (
    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
)
knude
\[\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(
    *knude, (t, -pi, pi), rendering_kw={"color": "blue"}, legend=False
)
../_images/28df684d7b889535ad3cb82aa85f9f3328e3da54fa2e51be84939db00cf3df1a.png
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7f2761316cd0>
## OBS -- Dette integral tager meget lant tid for SymPy. Kan nemt tage mere end et minut, så vær tålmodig
#integrate(V.subs({x: knude[0], y: knude[1], z: knude[2]}).dot(knude.diff(t)), (t, -pi, pi))

Det arbejde som kraftfeltet \(\pmb{V}\) bidrager med når partiklen kører en tur rundt i knuden er altså 0. Ganske som forventet, da \(\pmb{V}\) er et gradientvektorfelt.

Fladeintegral af vektorfelter (Flux gennem flader)#

Der er givet et vektorfelt ved:

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}\]

Vi betragter den del af standardparaboloiden \(F_{parab}\), der ligger under planen \(z=1\). Denne kan parametriseres ved:

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}\]

hvor \(u\in [0,1]\) og \(v\in [-\pi,\pi]\). Vektorfeltet og fladen illustreres ved:

p_parab = dtuplot.plot3d_parametric_surface(
    *r,
    (u, 0, 1),
    (v, -pi, pi),
    use_cm=False,
    camera={"elev": 25, "azim": -55},
    show=False
)
p_felt = 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_felt).show()
../_images/5a05960696cbaf905827eddf6d8f8d979d29e959ab4144398be3024ea3b168c7.png

Vi kan nøjes med at plotte vektorfeltet på selve fladen:

# Kommer vektorer fra alle hjørner. Der kommer for mange
# vektorer hvis "p_parab" bruges, så her sættes n ned
dummy_surface = dtuplot.plot3d_parametric_surface(
    *r, (u, 0, 1), (v, -pi, pi), n=8, show=False
)
p_felt2 = 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,
)
# bedre kamera vinkel til at se vektorerne her
p_parab.camera = {"elev": 60, "azim": -50}

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

Beregning af flux “ned” gennem paraboloiden \(F_{parab}\)#

Det ligner tydeligvis, at der er (positiv) flux op gennem fladen. Vi regner derfor med at finde at fluxen ned gennem fladen er negativ. Vi starter med at finde normalvektoren til parabloiden givet ved vores valgte parametrisering:

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}\]

Læg her mærke til, at normalvektor \(n\) peger i den forkerte retning (ind imod z-aksen og derfor med positiv \(z\)-koordinat) i forhold til at vi ønsker fluxen ned gennem fladen. Vi skal altså huske at skifte fortegn på normalvektoren \(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}\]

Længden er normalvektoren skal kun bruges til at argumentere for at parametriseringen er regulær:

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

Dette udtryk er positivt på det indre af \(\Gamma\), så det er den. Vi finder nu integranden:

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

Hvormed fluxen udregnes ved

(4)#\[\begin{equation} 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

Den ønskede flux er altså \(-\pi\)!

Beregning af fluxen gennem parabloidens “låg” \(F_{låg}\) i \(z\)-aksens (positive) retning#

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}\]

Retningen på normalvektoren passer den ønskede retning gennem fladen \(F_{låg}\).

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

Fluxen gennem låget er \(Flux(\pmb{V},F_{låg}) = 2\pi\).

Fluxen gennem den lukkede flade med parabloide og låg \(F_{lukket} = F_{parab} \cup F_{låg}\)#

p_låg = 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_feltlåg = dtuplot.plot_vector(
    V(x, y, z),
    (x, -1, 1),
    (y, -1, 1),
    (z, 0, 1),
    slice=p_låg[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_felt2 + p_låg + p_feltlåg).show()
../_images/845b8b7e8a8d0fda12010d8b5cd30b5dfbd30f1232cc5e2ecc10e7fc749a0ef3.png

Vi behøver dog ikke at regne yderligere! Vi har nemlig fluxen gennem alle fladerne, hvorved den samlede flux gennem den lukkede flade er \(Flux(\pmb{V},F_{lukket})=-\pi+2\pi=\pi\).

Flux gennem et stykke af en kugleflade#

Der er givet et vektorfelt,

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}\]

og et udsnit af en kegleflade,

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}\]

med \(u\in[\pi/6,\pi/2]\) og \(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_felt = 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_felt).show()
../_images/8f9434fa89270f774e8566990f4f8fe68289994c405abd02f435c39a8a6e49e4.png

Med fluxen udregnet til,

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 gennem et stykke af en kugleflade med Gauss (ikke-pensum)#

Vi sammenligner resultatet fra fluxen udregnet med Gauss’ sætning gennem en halvkugle med radius \(a\) (\(u\in[0,a]\), \(v\in[0,\pi/2]\) og \(w\in[-\pi,\pi]\)), med fluxen gennem dens to overflader. Det skulle meget gerne være det samme!

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}\]

Først ved Gauss’ sætning,

a = symbols("a", real=True, positive=True)
M = Matrix.hstack(r.diff(u), r.diff(v), r.diff(w))
# determinanten er positiv. Eneste led der ikke er
# kvadreret er sin(v), som er positiv indenfor v's grænser
jacobi = 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 * jacobi, (u, 0, a), (v, 0, pi / 2), (w, -pi, pi))
../_images/69683868b30b9e55b273a0231d623d79845dc99625877b38adc2fd9c776e3574.png

Og så gennem de to flader,

# Første flade tilsvarer at fastsætte u til radius
r1 = r.subs(u, a)
n1 = r1.diff(v).cross(r1.diff(w))  # har tjekket at den pejer udad
integrand1 = n1.dot(V.subs({x: r1[0], y: r1[1], z: r1[2]}))
flux1 = integrate(integrand1, (v, 0, pi / 2), (w, -pi, pi))
# Anden flade tilsvarer at fastsætte v til pi/2
# Tilbageværende bliver skiven med radius a
r2 = r.subs(v, pi / 2)
n2 = -r2.diff(u).cross(r2.diff(w))  # skal være negativ før at den pejer udad
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

Som forventet!

Flowkurver / Integralkurver#

Vektorfelter kan også “bestemme” en partikels bevægelse i planen eller rummet (flowkurve). Vi forestiller os at vi til tiden \(t=0\) smider en partikel ned i et kraftfelt i punktet \(\pmb{x}_0\), og vi ønsker at modellere partiklens bane (kurve) \(\pmb{r}(t)\) som funktion af tiden. Disse kurver kaldes for integral-kurver (og undertiden for flow-kurver). De er løsninger til differentialligningssystemet:

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

hvor \(\pmb{x}_0\) er begyndelsespunktet.

Vi betragter vektorfeltet

\[\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*}\]

og to punkter, der angiver to partikler startpunkt \(s_1\) og \(s_2\) i vektorfeltet \(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])

Kan vi nu finde flowkurverne disse to partikler. Vi angiver dem hhv. \(r_1\) og \(r_2\).

I dette tilfælde er differentialligningssystemet lineært, og vi kender løsningsmetoden fra Matematik 1a. Vi finder først systemmatricen:

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

Og vi kan bruge egenværdierne og egenvektoerne for \(A\) til at bestemme flowkurverne

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}\]

Både \(\verb|A.eigenvects()|\) og \(\verb|A.diagonalize()|\) giver det ønskede info, og hvilken man bruger er et preference spørgsmål.

Med \(\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}\]

Den generelle løsning stilles op som vi kender det fra systemer af førsteordens differentialligninger (med to forskellige egenværdier):

\[\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_generel = (
    k[0] * exp(ev[0][0] * u) * ev[0][-1][0] + k[1] * exp(ev[1][0] * u) * ev[1][-1][0]
)
r_generel
\[\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}\]

De angivne startpunkter er vores startbetingelser og ligningerne

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

løses for at finde vores specifikke flowkurver

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

Disse kan substitueres direkte ind i \(\boldsymbol r(u)\)

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

og vi får flowkurverne

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_felt = 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_felt + 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/09ef1dc217bf4d512778788c5d5121dac388c7a30a8c7ad08574b5e6240c717c.png

Løsning 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_generel = 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_generel
\[\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_generel.subs(u, 0) - s1)
ks_r2 = solve(r_generel.subs(u, 0) - s2)
ks_r1, ks_r2
../_images/600adb32cde75868873f340a360e62f91f99d24ee61a59e3cb5991cf6d9bc06e.png

og vi får samme flowkurver som ovenfor

r1 = r_generel.subs(ks_r1)
r2 = r_generel.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}\]

Et eksempel med Gauss-sætningen (ikke pensum)#

Givet vektorfeltet

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}\]

Samt den rummelige mængde \(\Omega\) bestemt ved

\[\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*} \]

Her er mængden illustreret:

a = symbols("a")
# Da man ikke kan plotte et 3D volumen (nemt), plotter vi overfladerne,
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_felt = 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_felt).show()

Vi vil nu bestemme fluxen af vektorfeltet \(\pmb{V}\) ud gennem den lukkede overflade af \(\Omega\) ved brug af gauss sætning!

Først finder vi parameterfremstillingen for det massive volumen,

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)

Vi kan tage determinanten, da

\[\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*}\]

Vi skal dog stadig tage absolut-værdien (se Definition 25.19). Dog kan man som regel få flottere udtryk at lade være med at tage absolutværdien, hvis man kan garantere at determinanten altid er positiv. Det er den i dette tilfælde, da \(v>0\), så

jacobian = M.det()
jacobian

Divergensen bliver nu

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

Divergensen begrænset til det parameteriserede område fåes altså til,

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

Da

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

får af Gauss’ sætning nu:

(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}\]

I dette eksempel er det særligt oplagt at benytte Gauss’ sætning, da vi både slipper for at udregne fluxen gennem \(5\) sideflader, hvorefter vi vil skulle ligge delresultaterne sammen, og da divergensen er matematisk simpel i forhold til sit vektorfelt. Dette skal dog vurderes hver gang en ny opgave stilles!