Demo 1: Continuity, Plotting, and Partial Derivatives#
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()
Welcome back after Christmas and January, and welcome to spring in Math1. There will be a whole lot of new curriculum, and among other things a whole lot of 3D plots! For this we have developed the Python library dtumathtools
, which will accompany you during the spring. It contains dtuplot
which should be used for plotting, as well as several good helper functions. You should have dtumathtools installed from Math1a; if not, please run the command
pip install -U dtumathtools
in your terminal.
Functions (of One Variable)#
We can define the function \(f: \mathbb{R} \to \mathbb{R}\), \(f(x)=x \operatorname{e}^x\) as a python function using the familiar command:
def f(x):
return x * exp(x)
The function is evaluated at the point \(x = -2\) with the command:
f(-2)
data:image/s3,"s3://crabby-images/9ef30/9ef30a819c7799fc3c50a40cd286633890e96752" alt="../_images/b6e1b3b9d8e202066faa6446674f62366158552c856671c27ccc663bd775c0e1.png"
Which has the numerical value:
f(-2).evalf()
data:image/s3,"s3://crabby-images/85856/85856cb530f572e0338a7c37c71f616c4e38a70f" alt="../_images/0444f753e37329b0613e8fff94ae9ccaa2eb1de475d9d23016b063931a0bcd06.png"
It is, however, rarely necessary to define functions with the def
command, and we will often work directly with the function expression:
x = symbols('x', real = True) # Necessary to use x as a symbolic variable (note the real=True, since R -> R)
f_expr = x * exp(x)
f_expr
data:image/s3,"s3://crabby-images/cb31c/cb31c3fe96cca6cd7a0b31c0e44e8b629673b80f" alt="../_images/3ae31244e67473a5ac8fc7ece285237016df4d71318e05f3b42e4411dd04e939.png"
Which is evaluated by:
f_expr.subs(x, -2)
data:image/s3,"s3://crabby-images/9ef30/9ef30a819c7799fc3c50a40cd286633890e96752" alt="../_images/b6e1b3b9d8e202066faa6446674f62366158552c856671c27ccc663bd775c0e1.png"
The function can be differentiated by:
f_prime = f_expr.diff(x)
f_prime
data:image/s3,"s3://crabby-images/14546/1454609af109366cf0dcb5fb4ea5359073170e10" alt="../_images/fd6bd66de492f2b223cfc4a096eef0ba3367ce8aaa240228948628894e0e113d.png"
We can also investigate the limits for \(x \to -\infty\), \(x \to \infty\) and \(x \to -2\) with the commands:
f_expr.limit(x, -oo), f_expr.limit(x, oo), f_expr.limit(x, -2)
data:image/s3,"s3://crabby-images/96300/963003a574d34186921e2ebda814a7d337fabb53" alt="../_images/23060d86505aca9cf523745af7a47c60d34e380346bff1b0fe404cc979edc179.png"
It should be no surprise that \(\lim_{x \to -2} f(x) = f(-2)\), since the function is continuous everywhere.
We can plot the function and its derivative by:
plot(f_expr, f_prime, (x, -5, 1))
data:image/s3,"s3://crabby-images/42d47/42d472cad1e636808541515efc293b41730e7452" alt="../_images/0f753bdb9fb0ed2f99749bbee40d6f39133746d2facab14bb628a6b84f08e887.png"
<sympy.plotting.backends.matplotlibbackend.matplotlib.MatplotlibBackend at 0x7fc12923f550>
A little more complicated example could be the piecewise-defined function \(g: \mathbb{R} \to \mathbb{R}\):
which in Python would be defined by:
def g(x):
if x < 0:
return -x
else:
return exp(x)
Using Sympy, we can define it as follows:
g_expr = Piecewise((-x, x < 0), (exp(x), x >= 0))
g_expr
which evaluates as expected:
g_expr.subs(x, -2)
data:image/s3,"s3://crabby-images/4ba2d/4ba2dffd11c5b73d6eb9f0e52e421ab44894f81d" alt="../_images/92f88f218e4707cb362e045ff538e4563ffc87ee097cc6f41c0154b31fb249ec.png"
We plot it using:
plot(g_expr,(x,-5,2), ylabel='g(x)')
data:image/s3,"s3://crabby-images/a9a00/a9a006c0ceaff97ef6b152a27ffa367902bce14c" alt="../_images/ff69387a9b9c3dd46266119461ed8cda5e0007f2dd7bad370d3d5f73c52936b1.png"
<sympy.plotting.backends.matplotlibbackend.matplotlib.MatplotlibBackend at 0x7fc12a464dc0>
Notice that the function is discontinuous at \(x = 0\). Be aware that Python/CAS cannot be used to prove discontinuity at \(x = 0\). This would require a so called epsilon-delta-argument, which Python/CAS cannot help us with. Sympy will, for example, differentiate the function without raising any suspicion of discontinuity.
g_expr.diff(x)
We should, however, note that the function isn’t differentiable at \(x = 0\), since it isn’t continuous at this point.
Partial Derivatives using diff
#
For functions of multiple variables we will introduce the partial derivative, as well as how they can be used. To showcase the partial derivatives we will consider the function:
x, y = symbols('x y')
f = x*y**2+x
f
data:image/s3,"s3://crabby-images/39ac4/39ac46eaeff5cfeb41c6a356fd888dd638bfe30b" alt="../_images/47ad93ddbee374ec0abaa43a424c1ada0278074ee6ec31a728f6906f737db93f.png"
which can be differentiated using diff
:
f.diff(x), f.diff(y)
data:image/s3,"s3://crabby-images/1ad9a/1ad9a50b58b554270a2ac6cf6bb5d30978854a99" alt="../_images/96cdb955fa422f69b2de23ee99067a79352c90072210d65e88d0b349fe1aee26.png"
Each of these expressions can be differentiated with respect to \(x\) and \(y\). Doing that would yield the following four functions:
f.diff(x).diff(x), f.diff(x).diff(y), f.diff(y).diff(x), f.diff(y).diff(y)
data:image/s3,"s3://crabby-images/e34ca/e34ca275ea3ed04ec215420831f603fc11662727" alt="../_images/0896bd82a49855d4c2a2bb44e08ac4a050c613164395f9383f164122b48fb026.png"
These are the second-order derivatives, and they can be calculated quicker as follows:
f.diff(x,2), f.diff(y,2), f.diff(x,y), f.diff(y,x)
data:image/s3,"s3://crabby-images/8870b/8870bdbe7cfcc1ce1ff7bb06c7abe8afccd3fe74" alt="../_images/eb575a275da7132aa8134dffece6afeef13b6639d4f483f9bbce8100bfce913e.png"
We can substitute in values for \(x\) and \(y\) and calculate, for example, \(\frac{\partial}{\partial x}f(x,y)\) evaluated at \((-2,3)\):
f.diff(x).subs({x:-2,y:3})
data:image/s3,"s3://crabby-images/39f0f/39f0f7e7242801ac4eb6334fc392e934dee674b6" alt="../_images/bb054c8c986ed7dab843f2368babfeb4b847485be1bc4d64c526942ba1d97770.png"
or \(\frac{\partial}{\partial x\partial y}f(x,y)\), evaluated at \((5,-13)\):
f.diff(x,y).subs({x:5,y:-13})
data:image/s3,"s3://crabby-images/02688/02688380d350a21b5dbe42dbe7896af1c06d43ee" alt="../_images/525da790f07f0ade39f9e268f9151abed4a5e2e5dc2551b651c54546a2cdda2c.png"
Plots#
Orienting Graphs#
We will now consider functions of multiple variables, hence plotting in 3D! This leaves us a decision to make, since we need to choose which angle we want to see the graphs from. If we don’t do anything dtuplot, will choose an angle for us, but if we wish to inspect the graph from a certain angle, then camera
can be used. Try changing the values for elev
and azim
.
f = 4-x**2-y**2
p=dtuplot.plot3d(f, (x,-3,3),(y,-3,3), camera = {"elev": 25, "azim": 45})
data:image/s3,"s3://crabby-images/feac4/feac49e88f3df4b24bc6b268680548bd622b7da1" alt="../_images/a0c34af871b5a0e78cbbe2cdd96d7e465c765749b5e3f62731e287cd06da0f90.png"
The plot above was generated as a static PNG-file, which is great if you wish to print or export the Notebook as a PDF. All plots will be static if we don’t do anything or if we use the command %matplotlib inline
.
If we instead run the command %matplotlib qt
(which in the following block has been commented out; try running the block after removing #
), then we enable interactive plots. All subsequent plots, will now “pop out” of VS Code, where we now can rotate the graph.
# %matplotlib qt
About Interactive Plots#
Note: %matplotlib qt
only works when running python on your own computer. It will, for instance, not work if you run Python on an online server like Google Colab. In such cases, widgets can be used instead %matplotlib ipympl
. This does, however, require you to install the package ipympl
.
Overview of commands:
# %matplotlib inline # static plots
# %matplotlib qt # QT (cute) interactive pop-out plots
# %matplotlib ipympl # Widget/ipynpl interactive inline plots (not as stable as QT, may require a restart of the kernel)
# %matplotlib --list # list of all backends
Aesthetics#
When we wish to change the aesthetics of our plots, we’ll need to use rendering_kw={...}
which may look intimidating. It just tells which rendering settings to use, for example color
, alpha
etc. In most cases it’s adequate to just write {...}
, which dtuplot will know means aesthetics.
dtuplot.plot3d(f, (x,-3,3),(y,-3,3), wireframe = True, rendering_kw = {"color": "red", "alpha": 0.5})
data:image/s3,"s3://crabby-images/0f53e/0f53eca01e3848551f5054950517c3a360819cb8" alt="../_images/b3ba44790fdf1a217b5faaf85dd44324e4285ba727182083c2d2d5430d11e40b.png"
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7fc1077c5880>
Some aesthetic choices are special enough to be outside of rendering_kw
, for example wireframe
as seen above, or use_cm
which denotes each value with a color, as seen below.
p=dtuplot.plot3d(f, (x,-3,3),(y,-3,3), use_cm=True, legend=True)
data:image/s3,"s3://crabby-images/73ce5/73ce581e20b8a0ba6b757ed5bc8d3c29307305ce" alt="../_images/adca93f71d72443ea4b69c653a03b5862d2751baf22ab688a9b32a79cec43139.png"
Contours#
We can also plot contours, i.e. a 2D plot of a 3D structure by:
dtuplot.plot_contour(f, (x,-3,3),(y,-3,3), is_filled=False)
data:image/s3,"s3://crabby-images/080a1/080a14fa197e50429018cddff7e12cfdc6e8bd56" alt="../_images/c9542a91aaa9c03973320cd809f1253e5ee129cb33920e1d5f6ff5a7bbd747da.png"
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7fc1075e0220>
And if we wish to choose the exact values shown:
zvals = [-2,-1,0,1]
dtuplot.plot_contour(f, (x,-3,3),(y,-3,3), rendering_kw={"levels":zvals, "alpha":0.5}, is_filled=False)
data:image/s3,"s3://crabby-images/48b91/48b91a194b19156590f810ce2eda6c8f24e4c73c" alt="../_images/27f7953644264be8c08db016e81b40e7511e046ef237e44b3eddc4a564c7bc75.png"
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7fc104c775e0>
Gradient Vector Fields#
Consider the vector field
The gradient of \(f\) can be computed as:
f = cos(x)+sin(y)
nf = Matrix([f.diff(x), f.diff(y)])
nf
The gradient can also be computed using dtutools.gradient
. Note that this function doesn’t tell us in which order the variables are differentiated.
dtutools.gradient(f)
The gradient vector field can be plotted as shown:
dtuplot.plot_vector(nf, (x,-pi/2,3/2*pi),(y,0,2*pi),scalar=False)
data:image/s3,"s3://crabby-images/bdc4c/bdc4c8d7e95c75a6a20ae4c452fdaa64184d6c1c" alt="../_images/45c390f25fec5ef29bc181c4c5453ea218e3dc389133deaaa5009962eb5e0f67.png"
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7fc104fe1a30>
Or if we want to get fancy (note that rendering_kw
has been split, such that the arrows and contours have their own list of aesthetics).
dtuplot.plot_vector(nf, (x,-pi/2,3/2*pi),(y,0,2*pi),
quiver_kw={"color":"black"},
contour_kw={"cmap": "Blues_r", "levels": 20},
grid=False, xlabel="x", ylabel="y",n=15)
data:image/s3,"s3://crabby-images/ac3f3/ac3f32908d22d422dcf69b40ee03da1c93cc194c" alt="../_images/2ec1d95235d8f2a2a3472ac82f7267aecabe4e2b9d4d795b25e5bfc2df1b7872.png"
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7fc104b02040>
Which in 3D can be visualised as
p = dtuplot.plot3d(f, (x,-pi/2,3/2*pi),(y,0,2*pi),use_cm=True, camera={"elev":45, "azim":-65}, legend=True)
data:image/s3,"s3://crabby-images/10363/10363c12958523fef9ae280e7193021840fdfb0e" alt="../_images/afc3ef5ef825f19b779ced18bfc4ca848a23fcad47c5929b56bb3c8166be7285.png"