Week 5: Taylor Approximations#

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

from sympy import *
from sympy.abc import x,y,z,u,v,w,t
from dtumathtools import *
init_printing()

Taylor Polynomials for Functions of one Variable#

We would like to try to approximate, respectively, \(\ln(x)\) and \(\sin(x)\) via Taylor polynomials and investigate how the degree of the polynomials influence the approximation of the original function.

The command for Taylor expansion in SymPy is series and it has the following format:

\(\verb|series(function, variable, x0, K+1)|\)

NOTE!!: It is important to remember that the number of terms \(K\) in SymPy’s function call must be one larger than the \(K\) that is described in the textbook. So, if one wants for instance an approximating polynomial of the sixth degree with the expansion point \(x_0 = 0\), meaning \(P_6(x)\), for the function \(\cos(x)\), one must in SymPy write:

series(cos(x), x, 0, 7)
../_images/6908c19c63b1f1cb123a15f7ed345c32a2d650b28584e54a4693224b0288eca8.png

Furthermore one can see that SymPy adds the term \(O(x^{K+1})\). This notation is called “Big O” and means, roughly speaking, that the error approaches zero faster than \(x^{N+1}\). So, it is a description of the remainder term \(R_K(x)\) as in Taylor’s formula 4.3.2 in the text book. It can, though, not be used for finding an expression of the remainder function. For that you must yourself make investigations on the function in the intented interval.

If we only want the Taylor polynomial (and not information about the size of the error), we will remove the \(O(x^{K+1})\) term using \(\verb|.removeO()|\):

series(cos(x), x, 0, 7).removeO()
../_images/19a4c93a3f5e3c8b5396d11ba95649901f2edb971176ae65aa7c804f154ae1e0.png

This is now a polynomial we can evaluate. So, with this information we can now investigate functions and their approximations. Consider for example the function \(f:=\ln(x)\):

Here we will first create a plot with \(\verb|show = False|\), in order for us to add the other plots to this same plot with \(\verb|.extend()|\). From our plot it is clear to see that when we increase \(K\), our approximation becomes better as well. We briefly check the same thing for \(\sin(x)\), this time with \(x_0 = 0\):

pl = plot(sin(x),xlim = (-3,3), ylim = (-3,3), show=False, legend = True)
for K in [0,1,2,3,6]:
    newseries = series(sin(x),x,0,K+1).removeO()
    display(Eq(Function(f'P_{K}')(x), newseries))
    newplot = plot(newseries,label = f"n = {K}", show=False)
    pl.extend(newplot)
pl.show()
../_images/3c4c0909ec706e190c9339dcbf430ffe0a5c6f2befe6f8315894cc05aee2aea2.png ../_images/c77cc46ac87690254c490cf88fcb5054cebc7d0b163d9dfff1f2d0e1e9ebe7a0.png ../_images/ac90744695feb4709f1e5e81a2388bdda6d7493e4781eb90ce44718029d9f281.png ../_images/6c4457d66351cb66a5d0986b7da7e962562457977e298a967ca5098bf20362a8.png ../_images/ead3e80052bcb00df866b20883f619c6f7ff346bdf2277e276f0cf0668b8b50c.png ../_images/b2ccc2c85385837334e01898899a23323d8dd4d30fed549d4e2f4ae37901c5e4.png

Here we can see that only four different lines are clearly visible. If one looks above the plot, it is clear why. By definition we know that

\[\begin{align*} P_0(x) &= f(x_0) = \sin(0) = 0 \\ P_1(x) &= f(x_0) + f'(x_0)(x-x_0)\\ &= \sin(0) + \cos(0) x \\ &= x \\ P_2(x) &= f(x_0) + f'(x_0)(x-x_0) + \frac{1}{2} f''(x_0)(x-x_0)^2 \\ &= \sin(0) + \cos(0) x - \sin(0) x^2 \\ &= x \end{align*}\]

for \(x\in\mathbb{R}\).

Evaluation of Remainder Function using Taylor’s Formula#

We want to try to find an approximate value of \(\ln\left(\frac{5}{4}\right)\) using the approximating polynomial \(P_3(x)\) expanded from the point \(x_0=1\).

We shall first determine \(P_3(x)\):

x = symbols("x")
P3 = series(ln(x),x,1,4).removeO()
P3
../_images/e823bdbf9c0d8b007b6e4706eb9af1225768443cf40c3cfb2782dceebafd0cb1.png
val = P3.subs(x,Rational(5/4))
val, val.evalf()
../_images/e36acbc749cc3bb6d06fd0aadb5ab54ab76eeaf79b73151f61e90284673fc45c.png

We konw from Taylor’s formula 4.3.1 that a \(\xi \in ]1;\frac{5}{4}[\) exists such that the error \(R_3(\frac{5}{4})\) can be written as:

\[\begin{equation*} R_3\left(\frac{5}{4}\right) = \frac{f^{(4)}(\xi)}{4!}\cdot\left(\frac{5}{4} - 1\right)^4. \end{equation*}\]

Here we shall first find out which number in the interval for \(\xi\) thta leads to the largest possible error. If we approximate our error with this, we can be certain that the true error is smaller. \(f^{(4)}(\xi)\) is the only term that is dependent on \(\xi\) and is determined to be

xi = symbols("xi")
diff(ln(x),x,4).subs(x,xi)
../_images/0197dd3f34f593d1e47c356fc7324e8274dbf6c40cf8b1984f56b8fe31c3d27e.png

Here we get the result \(-\frac{6}{\xi^4}\). Now we shall just analyse the expression to find out which \(\xi\) that makes the expressions largest possible. We can in this case see that \(\xi\) is found in the denominator, and the expression thus increases if \(\xi\) decreases. That means for this case that we simply have to choose the smallest value possible for \(\xi\), which in this case is 1. Now we can carry out our evaluation of the error.

R3 = abs(diff(ln(x),x,4).subs(x,1) * (5/4 - 1) ** 4 /(factorial(4)))
display(Eq(Function('R_3')(S('5/4')), R3))
print('The correct value of ln(5/4) is in the interval:')
Interval(val - R3, val + R3)
../_images/8e7a1ecf4d5d47070db0ee6a3b51e4f2dadfd1bfeea9de108ec4e99ddbe32c08.png
The correct value of ln(5/4) is in the interval:
../_images/a90a163f84a550c3ecc47518ecc9d796ff32d1928918213713ee10cfedaffbe2.png

We have now evaluated the error, and we can now garantee that the true value of \(\ln(\frac{5}{4})\) is within the interval \(]0.2229;0.2250[\), (open due to rounding). Let us compare this with Python’s value (which in itself is an approximation),

ln(5/4), Interval(val - R3, val + R3).contains(ln(5/4))
../_images/f87722e62c2cce6fcd22dc8972756e5ec5e636b0df173693dd709a294de9034c.png

Limit Values using Taylor’s Limit Formula#

We will now use Taylor’s limit formula to determine the limit value of different expressions. This is often usable for fractions where the numerator, denominator, or both contain expressions that are not easy to work with. The way to access Taylor’s limit formula in SymPy is via the function \(\text{series}\), where we avoid using \(\text{.removeO()}\) afterwards. In the textbook, epsilon functions are used, but SymPy uses the \(O\) symbol. The two concepts are not entirely comparable, but if SymPy writes \(O(x^{K+1})\), we can replace that by \(\varepsilon(x) \, x^{K}\), where \(\varepsilon(x) \to 0\) for \(x \to 0\).

Example 1#

We will first investigate the expression \(\frac{\sin(x)}{x}\) when \(x\rightarrow 0\),

series(sin(x),x,0,n=3)
../_images/9df087cf4fbb2905b5bfe241975ea953e706f581fb719d2b1f08b0a0c3b3591c.png

This gives us

\[\begin{gather*} \frac{\sin(x)}{x} = \frac{x + \epsilon (x) \cdot x^2}{x} = 1 + \epsilon (x)x \rightarrow 1, \text{ when } x \rightarrow 0 \end{gather*}\]

Example 2#

It can sometimes feel like guess work to figure out how many terms one needs to include, when one is to find a limit value using Taylor’s limit formula. Let us for example try

\[\begin{equation*} \frac{\mathrm e^x - \mathrm e^{-x} - 2x}{x-\sin(x)}, \end{equation*}\]

where \(x \rightarrow 0\). Let us evaluate the numerator and denominator separately and see what happens when we include more and more terms.

T = E ** x - E ** (-x) - 2*x
N = x - sin(x)
series(T,x,0,2), series(N,x,0,2)
../_images/38b482ee71ee51eb8f992d390f91e41cec4e5a5eb9b9db84c8e5618d6932283d.png

Too imprecise since we only get the remainter term.

series(T,x,0,3), series(N,x,0,3)
../_images/77b8d1262aa2706b6f6f667dfde5cdf075395ffec7a2472cc7337f093d7372c3.png

Still too imprecise.

series(T,x,0,4), series(N,x,0,4)
../_images/1dda725e5cc27e28c85a87e2e2b227a43d32ded0666671510fc17c853647d14b.png

Here we have something where both terms have something usable. Here we get:

\[\begin{gather*} \frac{\frac{x^3}{3}+\epsilon(x)x^3}{\frac{x^3}{6}+\epsilon(x)x^3} = \\ \frac{\frac{1}{3} + \epsilon(x)}{\frac{1}{6}+\epsilon(x)} \rightarrow \frac{\frac{1}{3}}{\frac{1}{6}} = 2, \text{ when } x \rightarrow 0 \end{gather*}\]

As a check we can use the \(\text{limit()}\) function

limit(T/N,x,0)
../_images/92f88f218e4707cb362e045ff538e4563ffc87ee097cc6f41c0154b31fb249ec.png

Taylor Polynomials for Functions of Two Variables#

We consider the following function of two variables:

\[ f:\mathbb{R}^2 \to \mathbb{R},\quad f(x,y) = \sin(x^2 + y^2). \]

It is plotted below:

x,y = symbols("x y", real = True)
f = sin(x ** 2 + y ** 2)
dtuplot.plot3d(f,(x,-1.5,1.5),(y,-1.5,1.5),rendering_kw={"color" : "blue"})
../_images/c07993808817978ee211eed086a2be873613c3ab856e745c4a4879de7b3e216f.png
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7f91f6f08b50>

Let us determine the approximating first-degree polynomial with expansion point \((0,0)\)

P1 = dtutools.taylor(f,[x,0,y,0],degree=2)
P1
../_images/651416cddb7ade65e68d31c70e7a3dbe2ef01977dfec97a06aaaa159fde31487.png
p = dtuplot.plot3d(P1,(x,-1.5,1.5),(y,-1.5,1.5),show=false,rendering_kw={"alpha" : 0.5},camera={"azim":-81,"elev":15})
p.extend(dtuplot.plot3d(f,(x,-1.5,1.5),(y,-1.5,1.5),show=False))
p.show()
../_images/9ec8050f00e83867bc2320608b537228d770215f035b60ca476d0c898fc6a54f.png

Here we can see that \(P1\) is located in the \((x,y)\) plane. With expansion point \((1/10,0)\) we have:

p = dtuplot.plot3d(dtutools.taylor(f,[x,0.1,y,0],degree=2),(x,-1.5,1.5),(y,-1.5,1.5),show=false,rendering_kw={"alpha" : 0.5},camera={"azim":-81,"elev":15})
p.extend(dtuplot.plot3d(f,(x,-1.5,1.5),(y,-1.5,1.5),show=False))
p.show()
../_images/16539b6b70d6a3ac412311568f12b3558f8040706e55711f8f9d63eb8b75b8d8.png

We will return to the expansion point \((0,0)\). Let us see how the approximating second-degree polynomial looks:

P2 = dtutools.taylor(f,[x,0,y,0],3)
P2
../_images/c0ac6529e1f861b45f59e6e16378f840c828f16f81a10660112701cb30520dfd.png
dtuplot.plot3d(f,P2,(x,-1.5,1.5),(y,-1.5,1.5))
../_images/fa60bda8045fa3baeb872c29cb7adc9a18de3bb7269e657017bf1903d23bef0d.png
<spb.backends.matplotlib.matplotlib.MatplotlibBackend at 0x7f91f447a730>

This time, the approximating polynomial is an elliptic paraboloid. Lastly, let us have a look at how the approximating sixth-degree polynomial looks:

P6 = dtutools.taylor(f,[x,0,y,0],7)
p = dtuplot.plot3d(f,(x,-1.5,1.5),(y,-1.5,1.5),show=False)
p.legend=True
p.show()
../_images/246af51ad395ef223c433668134bba2e52f9e197cad5ffe790457c785f631ba4.png

As expected, they are now much closer to each other. Let us investigate the error for these polynomials at different points to see how will they fit. Let us begin with \((0.2,0.2)\):

f_p1 = f.subs([(x, 1/5), (y, 1/5)])
P1_p1 = P1.subs([(x, 1/5), (y, 1/5)])
P2_p1 = P2.subs([(x, 1/5), (y, 1/5)])
P6_p1 = P6.subs([(x, 1/5), (y, 1/5)])

RHS_list = (f_p1 - P1_p1, f_p1 - P2_p1, f_p1 - P6_p1)
displayable_equations = [ Eq(Function(f'P_{i}')(S('1/5')), expression) for i, expression in zip((1,2,6), RHS_list) ]

display(*displayable_equations)
../_images/32c1dda94b2c9c211f14a0d5151d9f8bb2475671f8a6ffa2dfdbd2d44ffc6f82.png ../_images/faf91d7017c86e02d42ed2fc4e83470a13ecf4e59f48198b12ffe15c60d9ca57.png ../_images/7fcadc7eafc4ec5b8151a280a689a206d607287a66a3d2b083a6b910e1f321d7.png

It all looks right. The error is much smaller for the approximating polynomials of higher degrees. Let us try with \((0.5,0.5)\):

f_p2 = f.subs([(x,1/2),(y,1/2)])
P1_p2 = P1.subs([(x,1/2),(y,1/2)])
P2_p2 = P2.subs([(x,1/2),(y,1/2)])
P6_p2 = P6.subs([(x,1/2),(y,1/2)])

RHS_list = (f_p2 - P1_p2, f_p2 - P2_p2, f_p2 - P6_p2)
displayable_equations = [ Eq(Function(f'P_{i}')(S('1/5')), expression) for i, expression in zip((1,2,6), RHS_list) ]

display(*displayable_equations)
../_images/8abb836e11cf20a716ab6bb09349e6f4e787d77810e1011047048be80b28f600.png ../_images/48f15142f33d4ac124a05d452087f9bf8d7be6d367d9bae9a98187eeccedfe27.png ../_images/88694c06171a2dd80cd86d01bea73c320e5ad042e492579bf26bc983e15ec9b4.png

The farther away from the expansion point \((0,0)\) we go, the larger is an error must we expect.

(It should be mentioned that our comparisons are based on the assumption that SymPy’s own approximations are much better than ours. This is most likely quite a good assumption in this case, but it is important to know that SymPy, Maple, and all other computer tools also perform approximations.)

Taylor Polynomials for functions of Three Variables#

Consider the function:

\[\begin{equation*} f: \mathbb{R}^3\to \mathbb{R},\quad f(x_1,x_2,x_3) = \sin(x_1^2 - x_2)e^{x_3}. \end{equation*}\]

We wish to determine the second-degree Taylor polynomial with expansion point \(\boldsymbol{x}_0 = (1,1,0)\).

x1,x2,x3 = symbols('x1,x2,x3', real = True)
f = sin(x1**2 - x2)*exp(x3)
f
../_images/770beaef1f8fa3b14070211d17bfba37a174613189a7e01d34ad69d917d51a0a.png

The second-degree Taylor polynomial for a function of multiple variables is given by

\[\begin{equation*} P_2(\boldsymbol{x}) = f(\boldsymbol{x}_0) + \left<(\boldsymbol{x} - \boldsymbol{x}_0), \nabla f(\boldsymbol{x}_0)\right> + \frac{1}{2}\left<(\boldsymbol{x} - \boldsymbol{x}_0), \boldsymbol{H}_f(\boldsymbol{x}_0)(\boldsymbol{x}-\boldsymbol{x}_0)\right> \end{equation*}\]

First we define \(\boldsymbol{x}_0\) and \(\boldsymbol{x}\):

x = Matrix([x1,x2,x3])
x0 = Matrix([1,1,0])

Then we find \(\nabla f(\boldsymbol{x}_0)\) and \(\boldsymbol{H}_f(\boldsymbol{x}_0)\):

nabla_f = dtutools.gradient(f,(x1,x2,x3)).subs([(x1,x0[0]),(x2,x0[1]),(x3,x0[2])])
nabla_f
\[\begin{split}\displaystyle \left[\begin{matrix}2\\-1\\0\end{matrix}\right]\end{split}\]
Hf = dtutools.hessian(f,(x1,x2,x3)).subs([(x1,x0[0]),(x2,x0[1]),(x3,x0[2])])
Hf
\[\begin{split}\displaystyle \left[\begin{matrix}2 & 0 & 2\\0 & 0 & -1\\2 & -1 & 0\end{matrix}\right]\end{split}\]

Now \(P_2\) can be determined:

P2 = f.subs([(x1,x0[0]),(x2,x0[1]),(x3,x0[2])]) + nabla_f.dot(x - x0) + S('1/2')* (x - x0).dot(Hf*(x - x0))
P2.simplify()
../_images/e6acb93e0758b11d776bb27be7ddbf371376305eb8099834187a01e9f46db95f.png

We can now have a look at the difference between the approximating polynomial and the true function at some chosen values:

v1 = Matrix([1,1,0])
v2 = Matrix([1,0,1])
v3 = Matrix([0,1,1])
v4 = Matrix([1,2,3])
vs = [v1,v2,v3,v4]
for v in vs:
    print((f.subs({x1:v[0],x2:v[1],x3:v[2]}) - P2.subs({x1:v[0],x2:v[1],x3:v[2]})).evalf())
0
0.287355287178842
0.712644712821158
-12.9013965351501

Again we see that the error increases when we move farther away from the expansion point.