Week 7: Riemann Integrals in 1D and 2D#

Demo by Karl Johan Funch Måstrup Kristensen, Magnus Troen, and Jakob Lemvig

from sympy import *
from dtumathtools import *

init_printing()

You are now about to do integration. And in many circumstances, SymPy will be of good help. SymPy’s \(\verb|integrate()|\) function takes care of many integrals, but you will often experience moments when \(\verb|integrate()|\) has difficulties with an integral, in which case you must use your own knowledge from the course to make the expression more “edible” for SymPy. Sometimes SymPy needs it cut into smaller pieces to not suffocate.

Integration in One Dimension#

Very simply, anti-derivatives are found using SymPy with \(\verb|integrate(f, var)|\). \(\verb|var|\) can be ledt out in cases where the function only has one variable, but as soon as multiple variables appear (or symbols defined with \(\verb|symbols()|\)) one must define which variable to integrate to.

Consider for example a function \(f:\mathbb{R} \to \mathbb{R}\) given by

\[\begin{equation*} f(x) = e^x \sin(x). \end{equation*}\]

We find an anti-derivative

\[\begin{equation*} F(x) = \int f(x)\; dx = \int \mathrm e^x \sin(x)\; dx \end{equation*}\]
x = symbols("x", real=True)
f = sqrt(1 - x**2)
f
../_images/03ff3502eff8f99d8558ed69dba676dd75ec065699dc547f47a5aebcb570282f.png
F = integrate(f, x)  # Technically the x is implicit here so integrate(f) would also work
plot(f, F, legend=True)
../_images/9ad023d517adcea528e1ea6b1a0e8f9cb28a93ad704d3e1330a85820289730e3.png
<sympy.plotting.backends.matplotlibbackend.matplotlib.MatplotlibBackend at 0x7fe4693efbe0>

Notice, though, that SymPy does not add constant terms after it has integrated,

F
../_images/1521292d3ef7c70bef0f72507a12bd27ad888828c1c853c09a123aaed30bf03e.png

so a constant must be added afterwards manually if one in an exercise is asked to provide the indefinite integral,

C = symbols('C')

F = integrate(f, x) + C
F
../_images/5ee17ad56fb7a8c16e93e634920f699a4d2271d5f5a22b2b754e99c79da839d6.png

To find the definite integral

\[\begin{equation*} \int_{a}^b f(x)\; dx = F(b) - F(a), \end{equation*}\]

one can make use of the anti-derivative as shown above and/or let SymPy do it with \(\verb|integrate(f, (var, a, b))|\). For example, if we for the Riemann integral are given:

\[\begin{equation*} \int_{-1}^1 f(x)\; dx \end{equation*}\]
# Simulated calculation by hand using the fundamental theorem of calculus
display(F.subs(x, 1) - F.subs(x, -1))

# SymPy's integrate function
integrate(f, (x, -1, 1))
../_images/00ca8a80b7e56ef0b220abf0c45e63f6108988284bceed3472fe4cac7235fa74.png ../_images/00ca8a80b7e56ef0b220abf0c45e63f6108988284bceed3472fe4cac7235fa74.png

Integration in Multiple Dimensions#

For functions of multiple variables the definite integral

\[\begin{equation*} \int\int f(x_1, x_2)\; dx_1 \; dx_2 \end{equation*}\]

can be determined using SymPy with

integrate(f, x1, x2)

and for definite integrals, the limits are added in tuples just as for integration in one dimension:

\[\begin{equation*} \int_{a_1}^{b_1}\int_{a_2}^{b_2} f(x_1, x_2)\; dx_1 \; dx_2 \end{equation*}\]
integrate(f, (x1, a1, b1), (x2, a2, b2))

For example, have a look at the function

x1, x2 = symbols('x1 x2')
f2 = (x1*x2)**3 - x1**2 + sin(3 * x2) + x1*x2**2 + 4
f2
../_images/5ad31881249fe4a2305aae0940ad3517c06d6a2d5fbce9cb58ad991ccf3bbe4a.png
p = dtuplot.plot3d(f2, (x1, -1.2, 1.2), (x2, -1.2, 1.2), zlim=(-2, 6), use_cm=True, colorbar=False,
                   camera = {'azim': 25, 'elev': 20}, wireframe=True, show=False)

p.show()
../_images/53aa17309649377edfee0831c2b2d9eca6219b0350d01aac1094f99f3600f696.png

An anti-derivative can be found with

integrate(f2, x1, x2) + C
../_images/488f4eb6fdb13fc43c7564340c95fd144cc9475753ec08000cb2cbf6e7ebfc87.png

and a definite volume under the graph can be found using the definite integral

integrate(f2, (x1, -1.2, 1.2), (x2, -1.2, 1.2))
../_images/5b051784258972fbe8895e01d3e2743780ecda0e2e2e50d745e4ad7c8d9cee62.png

Tips and Tricks for making Sympy Integrate#

When SymPy runs into problems with an integral, here are some tricks that often will help.

Define Variables with Constraints#

State information about the variables you define. Is it for example a real variable? Then tell that to SymPy. The most frequent of such definitions are

  • real = True

  • positive = True

  • negative = True

  • nonnegative = True

  • nonzero = True

Some examples could be:

x1 = symbols('x1', real = True, nonnegative = True) # x1 is real and non-negative
x2 = symbols('x2', nonzero = True) # x2 is different from zero

Simplify before Integration#

Call simplify on the integrand before integration is carried out. This can in many cases reduce the time required for SymPy to integrate.

The work flow could look like this:

f = 1 / (x1 + x2)

integrand = simplify(f)
F = integrate(integrand, x1, x2)
F
../_images/af0e916dc29a259d088828ff07c3aa516fbe569acf215efccf9e5d196b87377b.png

Utilize the Linearity of Integrals#

Utilize the linearity of integrals. If you are to integrate the expression

\[\begin{equation*} \int_{c}^{d}\int_{a}^{b} \alpha f_1(x_1,x_2) + \beta f_2(x_1,x_2) \:\mathrm dx\,\mathrm dy, \end{equation*}\]

then it will often be to your advantage to work with

\end{equation*} \alpha \int_{c}^d\int_{a}^{b} f_1(x_1,x_2) ,\mathrm dx,\mathrm dy + \beta \int_{c}^{d}\int_{a}^{b} f_2(x_1,x_2),\mathrm dx,\mathrm dy. \end{equation*}

An example could be

a = 3
b = 10
c = 2
d = 5
f = S(27)/3 * cos(x1) - 4 * atan(x2 -x1)

integrand1 = simplify(cos(x1)) # Here it is not necessary to use simplify
integrand2 = simplify(atan(x2 - x1))

F = S(27)/3 * integrate(integrand1, (x1, a,b), (x2, c,d)) - 4 * integrate(integrand2, (x1, a,b), (x2, c,d))
F
../_images/5bf91c023c3cd488e9a69718f49c11d88cf35cb5ae0fe52c78f040029ff5fc59.png

Find an Anti-Derivative first and Insert Limits next#

Sometimes SymPy can find an anti-derivative but not evaluate the definite integral.

In such cases you can instead do the following:

f = tan(ln(x1**2))**3 / x1

integrand = simplify(tan(ln(x1**2))**3)

F = integrate(integrand, (x1,a,b))
F
../_images/6086e0193c3ff033a085bcc8fa16ebfda92bb8efda0b22763b177e2676f5a39f.png

We can instead find an anti-derivative and then insert the limits ourselves:

F = integrate(f, x1)

F.subs(x1, b) - F.subs(x1, a)
../_images/362d17659d49f436dfd9f5ae593051c410cf00ecf9efa2c91c011b1ee2291a36.png

.doit()#

If SymPy gives an output where the integration is not being carried out, one can in some cases force SymPy to evaluate the integral using

\[\begin{equation*} \verb|intragte(f, (x,a,b)).doit()| \end{equation*}\]

This risks giving an error, though, and then one must use one of the other methods.

Numerical Methods#

You can in some cases risk that SymPy either uses incredibly long time or cannot find a result for an integral at all, even after you have helped it out as much as you can. In such cases you might have no other choice than to solve the integral numerically. Methods for that require the numerical libraries numpy and scipy. You have already used those in Math1 as well as in your physics and programming courses, but if you have not installed them yet, then you can do that with:

or if one uses pip3

# quad is a numerical integrator for functions of one variable
# nquad is a numerical integrator for functions of multiple variables
from scipy.integrate import quad, nquad
import numpy as np

Below follows, respectively, an example of a numerical solution in one and in two dimensions.

One Dimension#

Let us have a look at the integral

\[\begin{equation*} \int_{\pi}^{10} \frac{\tan^3(\ln(x^2))}{x} \,\mathrm dx. \end{equation*}\]
f1 = tan(ln(x**2))**3 / x

# Note: SymPy won't determine the limited integral
integrate(f1, (x,pi,10))
../_images/361400984a0a84067c6b137e8cbf883e112491b21161515bb1a2b5fce939510b.png

This is one of those cases where SymPy is able to find an analytical solution if we just help out a bit. But let us begin by solving the integral numerically.

Firstly we will convert \(f_1\) to a python/lambda function. This is done with the function lambdify and corresponds, roughly speaking, to defining \(f_1\) by def f1(x):

f1_num = lambdify(x, f1, 'numpy')

# Now we can give f1_num a numerical value and get a numerical value back
f1_num(3)
../_images/c45bb55c23422385586a83def64f01d66bfcf5c0d521975eb04a3c277a4cb44c.png

Now, quad can be used to determine the integral

F1_num, error1 = quad(f1_num, np.pi, 10)
F1_num, error1
../_images/bbf00a475f0e046d0f4d6cee7136d029915c3e10977aa3a4cc9bc680ed5254b5.png

The function quad provides two outputs. The numerical approximation of the integral as well as an estimated error (deviation) from the exact solution.

We can try to compare this with the analytical solution. First we can force SymPy to evaluate the integral with \(\verb|.doit()|\), but this jsut results in an error:

# Remove # to see the error yourself. Maple

# integrate(f1, (x, pi, 10)).doit()

Instead we can use SymPy to determine an anti-derivative, and then ourselves compute the definite integral:

# ad: anti-derivative
F1_ad = integrate(f1, x)
F1_analytic = trigsimp(F1_ad.subs(x,10) - F1_ad.subs(x,pi))
F1_ad, F1_analytic
../_images/37db1dd826511d3523a7dc3f92494de793c3c3ca1c9ab7fde5ccd1464e2418e6.png

Let us compare the two results:

F1_analytic.evalf() - F1_num
../_images/a2f83ff867a8a98b48f5aa62f17d2618d9e4cf92e4268123a72158ed3bc8f26e.png

So, the numerical solution gives a fairly good approximation, but the two results are not entirely equal. Note that the true error that quad incurs is smaller than the estimated error of \(3.4 \cdot 10^{−11}\). And that is not bad.

Two Dimensions#

Let us have a look at the expression

\[\begin{equation*} \int_{0}^{\pi}\int_{-10}^{20} \sin(x)\cdot y^2\: \mathrm dy \,\mathrm dx. \end{equation*}\]

Here SymPy has no problems finding a solution, so we can easily compare the analytical to the numerical method.

x,y = symbols('x y', real = True)

f2 = sin(x)*y**2

F2_analytic = integrate(f2, (y,-10,20), (x,0,pi))
F2_analytic
../_images/8c76590505eded769f7a8bbdefc643ea2de2b147a56972c54fd3ec28b5702876.png

To evaluate the integral numerically we must again convert \(f_2\), using lambdify.

f2_num = lambdify((x,y), f2, 'numpy')   # lambdify can take multiple variables

# Now f2_num is a numerical function of two variables
f2_num(3,4)
../_images/0c351820de1d0aebec8ceca363148b11fe20cd34de28dc15b79daa53fa3ab2ea.png

Now the integral can be determined using nquad. For a function of \(n\) variables, nquad is called by:

\[\begin{equation*} \verb|nquad(func, [[range for var_1], [range for var_2], ..., [range for var_n]]|) \end{equation*}\]
F2_num, error2 = nquad(f2_num, [[0, np.pi], [-10,20]])
F2_num, error2
../_images/7a1b15d82326ef0424d6c85ad3ef69f0a332c96afc6cb941602381f0382c2f33.png

The two methods can now be compared:

F2_num - F2_analytic
../_images/651416cddb7ade65e68d31c70e7a3dbe2ef01977dfec97a06aaaa159fde31487.png