Uge 8: Riemann-Integraler i flere dimensioner og variabelskift#

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()

Python funktioner til parametriseringer#

Når vi arbejder med variableskift kan vi med fordel anvende at Python- og lambda-funktioner kan håndtere SymPy-objekter. For at illustrere hvordan dette kan anvendes ser vi på eksemplet

\[\begin{equation*} f: \mathbb{R}^3 \to \mathbb{R}, \quad f(x,y,z) = x+y+z \end{equation*}\]
\[\begin{equation*} \boldsymbol{r}(u,v,w) = \begin{bmatrix} u -v \\ w^2 \\ v(u+w) \end{bmatrix}, \quad u,v,w \in \mathbb{R} \end{equation*}\]

Vi vil gerne bestemme \(f(\boldsymbol{r}(u,v,w))\).

Til dette definere vi nu \(f\) som hhv. en SymPy-expression (variabel), en standard python-funktion og en lambda-funktion

x,y,z = symbols('x,y,z', real =True)
u,v,w = symbols('u,v,w', real =True)

# Parameterfunktion
r = Matrix([u -v, w**2, v*(u+w)])

# Funktioner
f_sym = x+y+z 

def f_function(x,y,z):
    return x+y+z

f_lam = lambda x,y,z : x+y+z

Som SymPy-expression kan \(f(\boldsymbol{r}(u,v,w))\) opnås ved

fr_sym = f_sym.subs(zip((x,y,z), r)).simplify()

Med Python- og lambda-funktionen kan vi i stedet slippe afsted med

fr_function = f_function(*r).simplify()
fr_lam = f_lam(*r).simplify()

Det er dog vigtigt at man husker *! Dette fortæller python at koordinaterne i \(\boldsymbol{r}\) skal gives til funktionen som individuelle argumenter

fr_sym, fr_function, fr_lam
../_images/b64799871e9911cbea4dfc52b8a8b85c8345cc2a21dd9ba099fc0db58be4cb19.png

Integration af områder begrænset af rette linjer#

Den simpleste parameterkurve er en ret linje. Et eksempel på dette er linjen mellem punkterne \((2,0)\) og \((4,4)\):

\[\begin{equation*} L := \Bigl\{(x,y) \in \mathbb{R}^2\ |\ 2\leq x \leq 4 \, \wedge \, y = 2x - 4\Bigr\} \end{equation*}\]

Parametriseret ved

\[\begin{equation*} \begin{bmatrix} x \\ y \end{bmatrix} = \begin{bmatrix} 2 \\ 0 \end{bmatrix} + u \begin{bmatrix} 2 \\ 4 \end{bmatrix}, \end{equation*} \]

hvor \(u \in [0,1]\). Når parameteren \(u\) gennemløber intervallet \([0,1]\) vil punktet \((x,y)\) travere linjen med start \((2,0)\) og slut \((4,4)\).

u,v = symbols("u v")
r_L = Matrix([2,0]) + Matrix([2,4]) * u
r_L
\[\begin{split}\displaystyle \left[\begin{matrix}2 u + 2\\4 u\end{matrix}\right]\end{split}\]
dtuplot.plot_parametric(*r_L, (u,0,1), xlim=(0,5), ylim=(0,5)) # *-tegnet foran r_l er vigtig at huske når man plotter parametriske kurver
../_images/1ddf7cfb0dd90108dcaa7057d1ae6d1d2d028aaef73ca9eaa7ab131903af2ed2.png
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7f7c6228f850>

Vi kan ofte gøre livet lidt nemmere, når vi skal parametrisere plane områder ved brug af linjestykker. Lad os kigge på to eksempler:

Eksempel#

Vi skal parametrisere området i planen begrænset af linjerne:

\(y = 1-x\)
\(y = 2x+1\)
\(x=2\)

som beskrevet matematisk er

\[\begin{equation*} \Omega:=\Bigl\{(x,y) \in \mathbb{R}^2\ \mid 0 \leq x \leq 2 \, \wedge \, 1 - x\leq y \leq 2x+1\Bigr\}. \end{equation*}\]

Lad os se hvordan det område ser ud:

x,y = symbols("x y")
område = dtuplot.plot_implicit(Eq(x,2),Eq(y,2*x+1),Eq(y,1-x),(x <= 2) & (y <= 2*x+1) & (y >= 1-x),(x,-0.1,5),(y,-1.1,5.1),aspect="equal",size=(8,8))
/builds/pgcs/pg1/venv/lib/python3.9/site-packages/spb/series.py:2214: UserWarning: The provided expression contains Boolean functions. In order to plot the expression, the algorithm automatically switched to an adaptive sampling.
../_images/4fdfe18ea338a937a5137122521501e2d546c30c90bb5b8f31441d645c5c8e6d.png

Vi vil gerne parametrisere hele området. Det betyder at alle linjer, der kan udspændes fra grafen \(y=-x + 1\) til grafen \(y=2x + 1\), skal beskrives. Vi beskriver punkterne ved bunden af pilen generelt med \(A = (u,1-u)\)
Toppen beskriver vi generelt med \(B = (u,2u+1)\)
Derfor er retningsvektoren \(AB\) beskrevet ved

\[\begin{equation*} B-A = \begin{bmatrix} u - u \\ 2u+1 - (1-u) \end{bmatrix} = \begin{bmatrix} 0 \\ 3u \end{bmatrix} \end{equation*}\]
AB = dtuplot.quiver(Matrix([1,0]),Matrix([0,3]),show=False,rendering_kw = {"color" : "black"})

område.extend(AB)
område.show()
../_images/1660b4ba8038b008ba50bce59120c0cba76cc86ff2a8946c333f4fef30f5170c.png

Vi kan så parametrisere et givet linjestykke mellem to punkter på toppen og bunden på følgende måde

\[\begin{equation*} \begin{bmatrix} x \\ y \end{bmatrix} = \begin{bmatrix} u \\ 1-u \end{bmatrix} + v\begin{bmatrix} 0 \\ 3u \end{bmatrix}, \, v \in [0,1]. \end{equation*} \]

Hvis vi vælger et \(u\), der løber i intervallet \([0,2]\), så vil vi gennemløbe alle linjerne fra venstre til højre, og på den måde dække hele trekanten.
Så hele område kan parametriseres som:

\[\begin{equation*} \begin{bmatrix} x \\ y \end{bmatrix} = \begin{bmatrix} u \\ 1-u +3vu\end{bmatrix}, \quad u \in [0,2], \, v \in [0,1]; \end{equation*}\]
dtuplot.plot3d_parametric_surface(u,1-u+3*u*v,0,(u,0,2),(v,0,1),camera={"elev":90, "azim":-90})
## Man kan ikke plotte planer i 2d, så vi bruger "camera" argumentet til at kigge på plottet fra oven
/builds/pgcs/pg1/venv/lib/python3.9/site-packages/spb/backends/matplotlib/matplotlib.py:599: UserWarning: Attempting to set identical low and high zlims makes transformation singular; automatically expanding.
../_images/8641d995b8ea83d96739de9d18139aa3f5bb95e1f4de3c775a5049b9d9d94315.png
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7f7c655b5eb0>

Områder begrænset af cirkler#

Vi vil nu parametrisere enhedscirkelskiven med centrum \((2,1)\). Fordi vi godt ved, hvordan en cirkel ser ud, så vi gå direkte efter af finde det linjestykke vi gerne vil parametrisere.

cirkel = dtuplot.plot_implicit(Eq((x-2) ** 2 + (y-1) ** 2,1),((x-2) ** 2 + (y-1) ** 2 <=1),(x,0,3),(y,0,3), aspect = 'equal',show=False)
AB = dtuplot.quiver(Matrix([2,1]),Matrix([cos(pi /4),sin(pi / 4)]),rendering_kw={"color":"black"}, show=False)
# cirkel.extend(AB)
(cirkel + AB).show()
../_images/b856c6ab3758d4e90fc24c88fb6663792038e55f80909680d16d01000cc705b5.png

Vi vil nu parametrisere enhver linje, der går fra centrum til et punkt på cirkelskivens rand. Dette gøres på følgende måde. Vores \(A\) er nu \((2,1)\), vores \(B\) er \((2 + \cos(u), 1 + \sin(u))\), så vores retningsvektor \(AB\) er derfor \(\begin{bmatrix} \cos(u) \\ \sin(u) \end{bmatrix}\). Denne specifikke linje parametriseres derfor som:

\[\begin{equation*} \begin{bmatrix} x \\ y \end{bmatrix} = \begin{bmatrix} 2 \\ 1 \end{bmatrix} + v \begin{bmatrix} \cos(u) \\ \sin(u) \end{bmatrix},\quad \text{for } \ v \in [0,1] \end{equation*}\]

Hvis vi lader \(u\) gennemløbe intervallet \([0,2 \pi]\), så kan vi få alle linjerne mellem centrum og randen, og på den måde få hele cirklen med. Så fladen parametriseres:

\[\begin{equation*} r_C(u) = \begin{bmatrix} 2 + v \cos(u) \\ 1 + v \sin(u) \end{bmatrix} = \begin{bmatrix} x \\ y \end{bmatrix},\quad \text{for } u \in [0,2\pi],\ v \in [0,1] \end{equation*}\]
rC = Matrix([2,1]) + v * Matrix([cos(u),sin(u)])
rC
\[\begin{split}\displaystyle \left[\begin{matrix}v \cos{\left(u \right)} + 2\\v \sin{\left(u \right)} + 1\end{matrix}\right]\end{split}\]
dtuplot.plot_parametric_region(*rC, (u,0,2*pi), (v,0,1), aspect="equal")
../_images/85d681e0006bb1bb7fc40ef4fbe2e7f2fec49275f8d68c1e3e77628612c4a6f6.png
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7f7c428cb370>

For hver af disse områder i planen kan et planintegral for en given funktion udregnes.

Planintegralet#

Givet funktionen

\[\begin{equation*} f(x,y) = 2xy \end{equation*}\]
f = lambda x,y: 2 * x * y
f(x,y)
../_images/5eb51609c18123e33c142bf33a160108162cf3a39ed02e93d5fd314f771dafad.png

ønsker vi finde planintegralet af \(f\) over området begrænset af vores cirkel, altså \(\int_C f(x,y)\; \mathrm{d}\pmb{x}\).

NB: Ved denne parameterfremstilling afbildes det akseparallelle kvadrat \([0, 2\pi] \times [0, 1]\) i \((u,v)\)-planen over i den cirkel vi integrerer over i \((x,y)\)-planen.

dtuplot.plot_implicit(((x < 2*pi) & (x > 0) & (y > 0) & (y < 1)), (x, -0.5, 2*pi + 0.5), (y, -0.5,1.5),
                      title="uv plan", xlabel="u", ylabel="v", axis_center='auto')

print("afbilledes i over")

cirkel.show()
../_images/df5a93f2847793a3a7c8b0ea6ca707c2d6191bf4671ec6731d1f8bddf20f2519.png
afbilledes i over
../_images/6e92011807954c9d09c335a8eacfd06b08da1f3576b36b48f9ff5c15b586155a.png

Nu får vi brug for absolut-værdien af Jacobi-determinanten, som lokalt måler, hvor meget \((u,v)\)-parameterområdet deformeres, når det udsættes for afbildningen \(\boldsymbol r_C\). Og da der nu er flere variable og afledede, opstiller vi først Jacobimatricen:

\[\begin{equation*} \boldsymbol{J}_{r_C} = \left[\begin{matrix} \vdots & \vdots \\ \frac{\partial \boldsymbol r(u,v)}{\partial u} & \frac{\partial \boldsymbol r(u,v)}{\partial v} \\ \vdots & \vdots \end{matrix}\right] = \left[\begin{matrix}\nabla(2 + v\cos\left(u \right))^T\\ \nabla(1 + v\cos\left(u \right))^T\end{matrix}\right] \end{equation*}\]
JacobiM = Matrix.hstack(rC.diff(u), rC.diff(v))
JacobiM
\[\begin{split}\displaystyle \left[\begin{matrix}- v \sin{\left(u \right)} & \cos{\left(u \right)}\\v \cos{\left(u \right)} & \sin{\left(u \right)}\end{matrix}\right]\end{split}\]

Nu kan vi finde absolut-værdien af Jacobi-determinanten \(|\det(\boldsymbol J_{r_C})|\) ved

Jacobiant = abs(det(JacobiM)).simplify()
Jacobiant
../_images/f67aaaf220cda25526306bbfa832ab57b69a74fdb8a7d71f8441be12bf2ffd86.png

Integralet af \(f\) over \(C\) er da ifølge Transformationsætningen givet ved:

\[\begin{equation*} \int_C f(x,y)\; \mathrm{d}\pmb{x} = \int_{0}^{1} \int_{0}^{2\pi} f(\boldsymbol r_C(u,v))\cdot|\det (\boldsymbol J_{r_C}) |\;\mathrm{d}u\mathrm{d}v = \int_{0}^{1} \int_{0}^{2\pi} f(2+v\cos(u),1+v\sin(u))\cdot|v|\; \mathrm{d}u \mathrm{d}v \end{equation*}\]
integrate(f(*rC) * Jacobiant,(u,0,2*pi),(v,0,1))
../_images/0dc6acaeffe8f6d4d8815772fcbfeba3f65aed62fd03bd46256545d25675d4a6.png

Rumintegral: Vægten af en kugle i \(\mathbb{R}^3\)#

Lad

\[\begin{equation*} \Omega = \left\{\boldsymbol{x} \in \mathbb{R}^3 \:|\: ||\boldsymbol{x}||_2 \leq 1 \right\} \end{equation*}\]

Altså beskriver \(\Omega\) en enheds kugle i \(\mathbb{R}^3\). Lad nu

\[\begin{equation*} f:\mathbb{R}^3 \to \mathbb{R}, \quad f(\boldsymbol{x}) = 1 \end{equation*}\]

beskrive en konstant “massetæthedsfunktion” (alle dele af kuglen antages altså at veje lige meget). Vi ønsker at bestemme vægten af kuglen beskrevet af \(\Omega\) altså skal vi bestemme

\[\begin{equation*} M =\int_{\Omega} f\; \mathrm{d}\pmb{x} \end{equation*}\]

Dette kræver at vi først og fremmest skal bestemme en parametrisering for kuglen, og den tilhørende jacobi-determinant. Den nemmeste ville være at benytte sfæriske koordinater som beskrevet i bogen, men vi vil prøve at arbejde os frem til en parametrisering ud fra hvad vi ved om parametriseringen af a cirklen.

Parametrisering af en kugle#

u,v,w = symbols('u v w', real = True)

Vi vil anvende den tilgang at en kugle kan beskrives som en cirkel i xz-planen roteret 180 grader om z-aksen. Cirklen af parametriseret således:

\[\begin{equation*} \boldsymbol{r}_{cirkel} = \begin{bmatrix} \cos(u)\\ 0\\ \sin(u) \end{bmatrix}, \quad u\in[0,2\pi] \end{equation*}\]
r_circle = Matrix([cos(u), 0, sin(u)])

dtuplot.plot_parametric(r_circle[0], r_circle[2], (u,0,2*pi), use_cm=False,aspect="equal", xlabel = 'x', ylabel = 'z')
../_images/b0327b7f0e774fdc16ac2849e9ebae654c9b9479551942de3137b0cb404ebcbd.png
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7f7c424bcd30>

Herefter kan cirklen roteres om z-aksen vha rotationsmatricen

\[\begin{equation*} R_z = \left[\begin{matrix}\cos{\left(v \right)} & - \sin{\left(v \right)} & 0\\\sin{\left(v \right)} & \cos{\left(v \right)} & 0\\0 & 0 & 1\end{matrix}\right], \quad v\in[0,\pi] \end{equation*}\]

Dette vil altså give en sfære med radius 1.

Rz = Matrix([[cos(v), -sin(v), 0], [sin(v), cos(v), 0], [0, 0, 1]])

r_sphere = simplify(Rz * r_circle)
r_sphere
\[\begin{split}\displaystyle \left[\begin{matrix}\cos{\left(u \right)} \cos{\left(v \right)}\\\sin{\left(v \right)} \cos{\left(u \right)}\\\sin{\left(u \right)}\end{matrix}\right]\end{split}\]

Tilsidst kan vi opnå en solid kugle ved at indføre parameteren \(w \in [0,1]\), som sørger for at vi kan ramme alle radier mellem 0 og 1. Altså er den parameterfremstillingen for kuglen:

\[\begin{equation*} \boldsymbol{r}_{ball} = \left[\begin{matrix}w\cos{\left(u \right)} \cos{\left(v \right)} \\ w\cos{\left(u \right)}\sin{\left(v \right)}\\ w\sin{\left(u \right)}\end{matrix}\right], \quad u\in[0,2\pi], v\in[0,\pi], w\in[0,1] \end{equation*}\]
r_ball = r_sphere * w
r_ball
\[\begin{split}\displaystyle \left[\begin{matrix}w \cos{\left(u \right)} \cos{\left(v \right)}\\w \sin{\left(v \right)} \cos{\left(u \right)}\\w \sin{\left(u \right)}\end{matrix}\right]\end{split}\]

Dette kan visualiseres ved at kigge på \(\boldsymbol{r}_{ball}\) med forskellige værdier for \(w\)

big_ball = dtuplot.plot3d_parametric_surface(*r_ball.subs(w,1), (u, 0, 2*pi), (v, 0, pi), aspect = 'equal', label = 'w=1', rendering_kw = {'alpha': 0.4,}, show = false)
half_ball = dtuplot.plot3d_parametric_surface(*r_ball.subs(w,0.5), (u, 0, 2*pi), (v, 0, pi),label ='w = 0.5', rendering_kw = {'alpha': 0.5}, show = False)
quarter_ball = dtuplot.plot3d_parametric_surface(*r_ball.subs(w,0.25), (u, 0, 2*pi), (v, 0, pi),label ='w = 0.25', show = False)
(big_ball + half_ball + quarter_ball).show()
../_images/07cb9da862d06875d76d55ae876f0aed79b34a28ecfa4703cf75ad7387bf81bd.png

Jacobi-determinanten og tæthedsfunktion#

Jacobi-determinanten \(\det(\boldsymbol{J}_{r_{ball}})\) findes ved

\[\begin{equation*} \boldsymbol{J}_{r_{ball}} = \left[\begin{matrix} \vdots & \vdots & \vdots \\ \frac{\partial \boldsymbol r_{ball}(u,v,w)}{\partial u} & \frac{\partial \boldsymbol r_{ball}(u,v,w)}{\partial v} & \frac{\partial \boldsymbol r_{ball}(u,v,w)}{\partial w} \\ \vdots & \vdots & \vdots \end{matrix}\right] = \left[\begin{matrix}\nabla(w\cos{\left(u \right)} \cos{\left(v \right)})^T\\ \nabla(w \cos{\left(u \right)}\sin{\left(v \right)})^T\\ \nabla(w\sin{\left(u \right)})^T\end{matrix}\right] \end{equation*}\]

Dette kan let gøres med dtumathtools, hvor vi medtager absolut-værdien:

J_ball = r_ball.jacobian([u,v,w])
J_ball
\[\begin{split}\displaystyle \left[\begin{matrix}- w \sin{\left(u \right)} \cos{\left(v \right)} & - w \sin{\left(v \right)} \cos{\left(u \right)} & \cos{\left(u \right)} \cos{\left(v \right)}\\- w \sin{\left(u \right)} \sin{\left(v \right)} & w \cos{\left(u \right)} \cos{\left(v \right)} & \sin{\left(v \right)} \cos{\left(u \right)}\\w \cos{\left(u \right)} & 0 & \sin{\left(u \right)}\end{matrix}\right]\end{split}\]
Jacobiant = abs(det(J_ball)).simplify()
Jacobiant
../_images/7f5f2fcd6de3ec9e68029a1ca71d8db8ff0c59503f0b0dd4f8f7d69a72e28858.png

Tæthedsfunktionen er i dette tilfælde 1-funktionen og derfor ikke nødvendig at medregne, men lad os definere den for formalitetens skyld

f = lambda x,y,z: 1

Bestemmelse af integralet#

Nu kan vi bestemme vægten af kuglen ved

\[\begin{equation*} M =\int_{\Omega} f\ \mathrm{d}\pmb{x} = \int_{0}^{1} \int_{0}^{\pi}\int_{0}^{2\pi} f(\boldsymbol{r}_{ball})J_{r_{ball}} \;dudvdw \end{equation*}\]

Først definerer og simplificerer vi integranden

integrand = (f(*r_ball) * Jacobiant).simplify()
integrand
../_images/7f5f2fcd6de3ec9e68029a1ca71d8db8ff0c59503f0b0dd4f8f7d69a72e28858.png
M = integrate(integrand, (u, 0, 2*pi), (v, 0, pi), (w, 0, 1))
M
../_images/7be2cd03b8720c2cf711d8fde4c36bd142d1d8847983179c406b5e04d471aea2.png

Bemærk at \(M\) er lig med volumet af en kugle med radius 1, hvilket giver god mening da alle punkter er blevet tildelt vægten \(1\).

En anden tæthedsfunktion#

Lad nu

\[\begin{equation*} f_1 : \mathbb{R}^3 \to \mathbb{R}, \quad f_1(\boldsymbol{x}) = 2||x||_2 \end{equation*}\]

beskrive en ny tæthedsfunktion.

Lad os bestemme

\[\begin{equation*} M_1 = \int_{\Omega} f_1 d\pmb{x} \end{equation*}\]
f1 = lambda x,y,z: 2*sqrt(x**2 + y**2 + z**2)
integrand = (f1(*r_ball) * Jacobiant).simplify()
integrand
../_images/5505d7dfd8c3400cc827bd7fdd4d78e567212295d4f309014bd0b251466d20c5.png
M1 = integrate(integrand, (u, 0, 2*pi), (v, 0, pi), (w, 0, 1))
M1
../_images/46c7d195b5b0b7a95cd9748c4760d952374bc08b95fe0c21e993f68d5eedb159.png