In Numerical integration we derived methods of order 2, 4, and higher for numerical integration. They all use a formula
∫ − 1 1 f ( x ) d x ≈ ∑ k = 0 n w k f ( t k ) \int_{-1}^1 f(x)\, dx \approx \sum_{k=0}^n w_k f(t_k) ∫ − 1 1 f ( x ) d x ≈ k = 0 ∑ n w k f ( t k ) for a collection of nodes t 0 , … , t n t_0,\ldots,t_n t 0 , … , t n in [ − 1 , 1 ] [-1,1] [ − 1 , 1 ] and weights w 0 , … , w n w_0,\ldots,w_n w 0 , … , w n . (Throughout this section we use [ − 1 , 1 ] [-1,1] [ − 1 , 1 ] as the domain of the integral; for a general interval [ a , b ] [a,b] [ a , b ] , see Exercise 4 .) The nodes and weights are independent of the integrand f ( x ) f(x) f ( x ) and determine the implementation and properties of the formula.
The process for deriving a specific method was to interpolate the integrand, then integrate the interpolant. Piecewise linear interpolation at equally spaced nodes, for instance, produces the trapezoid formula. When the integrand is approximated by a spectrally accurate global function, the integration formulas are also spectrally accurate.
9.6.1 Periodic functions ¶ For a function periodic on [ − 1 , 1 ] [-1,1] [ − 1 , 1 ] , the most natural interpolant is the trigonometric polynomial (9.5.4) . However, from (9.5.3) one finds that
∫ − 1 1 ∑ k = − n n y k τ k ( x ) d x = ∑ k = − n n y k [ ∫ − 1 1 τ k ( x ) d x ] = 2 2 n + 1 ∑ k = − n n y k . \int_{-1}^1 \sum_{k=-n}^n y_k\tau_k(x)\, dx = \sum_{k=-n}^n y_k\left[ \int_{-1}^1 \tau_k(x)\, dx\right] = \frac{2}{2n+1} \sum_{k=-n}^n y_k. ∫ − 1 1 k = − n ∑ n y k τ k ( x ) d x = k = − n ∑ n y k [ ∫ − 1 1 τ k ( x ) d x ] = 2 n + 1 2 k = − n ∑ n y k . In Exercise 1 you are asked to verify that this result is identical to the value of the trapezoid formula on 2 n + 1 2n+1 2 n + 1 nodes.
9.6.2 Clenshaw–Curtis integration ¶ Suppose f f f is smooth but not periodic. If we use a global polynomial interpolating f f f at the Chebyshev second-kind points from (9.3.2) ,
t k = − cos ( k π n ) , k = 0 , … , n , %:label: chebextremerepeat
t_k = - \cos\left(\frac{k \pi}{n}\right), \qquad k=0,\ldots,n, t k = − cos ( n kπ ) , k = 0 , … , n , and integrate the resulting polynomial interpolant, the method should have spectral accuracy for a smooth integrand. The resulting algorithm is known as Clenshaw–Curtis integration .
Having specified the nodes in (9.6.1) , all that remains is to find the weights. The Lagrange form of the interpolating polynomial is
p ( x ) = ∑ k = 0 n f ( x k ) ℓ k ( x ) . p(x) = \sum_{k=0}^{n} f(x_k) \ell_k(x). p ( x ) = k = 0 ∑ n f ( x k ) ℓ k ( x ) . From this,
I = ∫ − 1 1 f ( x ) d x ≈ ∫ − 1 1 p ( x ) d x = ∫ − 1 1 ∑ k = 0 n f ( x k ) ℓ k ( x ) d x = ∑ k = 0 n f ( x k ) ∫ − 1 1 ℓ k ( x ) d x = ∑ k = 0 n w k f ( x k ) , \begin{align*}
I = \int_{-1}^1 f(x)\, dx \approx \int_{-1}^1 p(x) \, d x &= \int_{-1}^1 \sum_{k=0}^n f(x_k) \ell_k(x) \, d x\\
&= \sum_{k=0}^n f(x_k) \int_{-1}^1 \ell_k(x) \, d x = \sum_{k=0}^n w_k f(x_k),
\end{align*} I = ∫ − 1 1 f ( x ) d x ≈ ∫ − 1 1 p ( x ) d x = ∫ − 1 1 k = 0 ∑ n f ( x k ) ℓ k ( x ) d x = k = 0 ∑ n f ( x k ) ∫ − 1 1 ℓ k ( x ) d x = k = 0 ∑ n w k f ( x k ) , where w k = ∫ − 1 1 ℓ k ( x ) d x . w_k = \int_{-1}^1 \ell_k(x)\,dx. w k = ∫ − 1 1 ℓ k ( x ) d x . For even values of n n n the result is
w k = { 1 n 2 − 1 , k = 0 or k = n , 4 n ∑ j = 0 n / 2 cos ( 2 π j k / n ) γ j ( 1 − 4 j 2 ) , k = 1 , … , n − 1 , γ j = { 2 , j = 0 or n / 2 , 1 , j = 1 , 2 , … , n / 2 − 1. w_k =
\begin{cases}
\dfrac{1}{n^2-1}, & \text{$k=0$ or $k=n$},\\[3mm]
\dfrac{4}{n} \displaystyle \sum_{j=0}^{n/2} \frac{\cos ( 2 \pi j k / n)}{\gamma_j
(1-4 j^2) }, & k=1,\dots,n-1,
\end{cases}
\quad
\gamma_j =
\begin{cases}
2, & j=0 \text{ or } n/2,\\
1, & j=1,2,\dots,n/2-1.
\end{cases} w k = ⎩ ⎨ ⎧ n 2 − 1 1 , n 4 j = 0 ∑ n /2 γ j ( 1 − 4 j 2 ) cos ( 2 πjk / n ) , k = 0 or k = n , k = 1 , … , n − 1 , γ j = { 2 , 1 , j = 0 or n /2 , j = 1 , 2 , … , n /2 − 1. There are different formulas for odd values of n n n . Note that the weights also depend on n n n ; e. g. w 2 w_2 w 2 for n = 4 n=4 n = 4 is not the same as w 2 w_2 w 2 for n = 10 n=10 n = 10 . Also note that the interpolant itself never needs to be computed.
Function 9.6.1 performs Clenshaw–Curtis integration for even values of n n n .[1]
Clenshaw–Curtis integration
Clenshaw-Curtis integration1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
function [I,x] = ccint(f,n)
% CCINT Clenshaw-Curtis numerical integration.
% Input:
% f integrand (function)
% n one less than the number of nodes (even integer)
% Output:
% I estimate of integral(f,-1,1)
% x evaluation nodes of f (vector)
% Find Chebyshev extreme nodes.
theta = pi*(0:n)'/n;
x = -cos(theta);
% Compute the C-C weights.
c = zeros(1,n+1);
c([1 n+1]) = 1/(n^2-1);
theta = theta(2:n);
v = ones(n-1,1);
for k = 1:n/2-1
v = v - 2*cos(2*k*theta)/(4*k^2-1);
end
v = v - cos(n*theta)/(n^2-1);
c(2:n) = 2*v/n;
% Evaluate integrand and integral.
I = c*f(x); % use vector inner product
Clenshaw–Curtis integration1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
def ccint(f, n):
"""
ccint(f, n)
Perform Clenshaw-Curtis integration for the function f on n+1 nodes in [-1,1]. Return
integral and a vector of the nodes used. Note: n must be even.
"""
# Find Chebyshev extreme nodes.
theta = np.linspace(0, np.pi, n + 1)
x = -np.cos(theta)
# Compute the C-C weights.
c = np.zeros(n + 1)
c[[0, n]] = 1 / (n**2 - 1)
v = np.ones(n - 1)
for k in range(1, int(n / 2)):
v -= 2 * np.cos(2 * k * theta[1:-1]) / (4 * k**2 - 1)
v -= np.cos(n * theta[1:-1]) / (n**2 - 1)
c[1:-1] = 2 * v / n
# Evaluate integrand and integral.
I = np.dot(c, f(x)) # use vector inner product
return I, x
9.6.3 Gauss–Legendre integration ¶ Let us reconsider the generic numerical integration formula (9.6.1) ,
∫ − 1 1 f ( x ) d x ≈ ∑ k = 1 n w k f ( t k ) = Q n [ f ] , \int_{-1}^1 f(x)\, dx \approx \sum_{k=1}^n w_k f(t_k) = Q_{n}[f], ∫ − 1 1 f ( x ) d x ≈ k = 1 ∑ n w k f ( t k ) = Q n [ f ] , where Q n [ f ] Q_n[f] Q n [ f ] stands for the application of the formula to function f f f . (We start the sum from k = 1 k=1 k = 1 instead of k = 0 k=0 k = 0 for notational convenience in what follows.)
The interpolation approach spurred us to use Chebyshev nodes. But it’s not clear that these are ideal nodes for the specific application of finding an integral. Instead, we can define formula as the integral of a polynomial interpolant, but with the weights and nodes chosen to satisfy an optimality criterion. As usual, we denote the set of all polynomials of degree at most m m m by P m \mathcal{P}_m P m .
Since there are n n n nodes and n n n weights available to choose, it seems plausible to expect m = 2 n − 1 m=2n-1 m = 2 n − 1 , and this intuition turns out to be correct. Hence the goal is now to find nodes t k t_k t k and weights w k w_k w k such that
∫ − 1 1 p ( x ) d x = Q n [ p ] = ∑ k = 1 n w k p ( t k ) , p ∈ P 2 n − 1 . \int_{-1}^1 p(x)\,dx = Q_{n}[p] = \sum_{k=1}^n w_k p(t_k), \qquad p \in \mathcal{P}_{2n-1}. ∫ − 1 1 p ( x ) d x = Q n [ p ] = k = 1 ∑ n w k p ( t k ) , p ∈ P 2 n − 1 . If these conditions are satisfied, the resulting method is called Gauss–Legendre integration or simply Gaussian integration . Because the integration formula is linear, i.e., Q n [ α p + q ] = α Q n [ p ] + Q n [ q ] Q_n[\alpha p + q] = \alpha Q_n[p] + Q_n[q] Q n [ α p + q ] = α Q n [ p ] + Q n [ q ] , it is sufficient to show that Q n Q_n Q n gets the exact value for the monomials 1 , x , x 2 , … , x 2 n − 1 . 1,x,x^2,\ldots,x^{2n-1}. 1 , x , x 2 , … , x 2 n − 1 .
As an example, consider the case n = 2 n=2 n = 2 . Applying the integration formula to each monomial of degree less than 2 n 2n 2 n , we get the conditions
2 = w 1 + w 2 0 = w 1 t 1 + w 2 t 2 2 3 = w 1 t 1 2 + w 2 t 2 2 0 = w 1 t 1 3 + w 2 t 2 3 . \begin{split}
2 &= w_1 + w_2\\
0 &= w_1 t_1 + w_2 t_2 \\
\frac{2}{3} &= w_1 t_1^2 + w_2 t_2^2\\
0 &= w_1 t_1^3 + w_2 t_2^3.
\end{split} 2 0 3 2 0 = w 1 + w 2 = w 1 t 1 + w 2 t 2 = w 1 t 1 2 + w 2 t 2 2 = w 1 t 1 3 + w 2 t 2 3 . These equations can be solved to obtain
w 1 = w 2 = 1 , t 1 = − 1 3 , t 2 = 1 3 , w_1=w_2=1, \quad t_1 = -\frac{1}{\sqrt{3}}, \quad t_2 =
\frac{1}{\sqrt{3}}, w 1 = w 2 = 1 , t 1 = − 3 1 , t 2 = 3 1 , which specifies the two-point Gaussian integration formula.
Generalizing the process above to general n n n would be daunting, as the conditions on the nodes and weights are nonlinear. Fortunately, a more elegant approach is possible.
Choose an arbitrary p ∈ P 2 n − 1 p\in\mathcal{P}_{2n-1} p ∈ P 2 n − 1 , and let p ^ n ( x ) \hat{p}_n(x) p ^ n ( x ) be the lowest-degree interpolating polynomial for p p p using the as-yet unknown nodes t 1 , … , t n t_1,\dots,t_n t 1 , … , t n . By definition,
Q n [ p ] = ∫ − 1 1 p ^ n ( x ) d x . Q_n[p] = \int_{-1}^1 \hat{p}_n(x)\, dx. Q n [ p ] = ∫ − 1 1 p ^ n ( x ) d x . Since p ^ n ( x ) \hat{p}_n(x) p ^ n ( x ) has degree at most n − 1 n-1 n − 1 , it is exactly equal to p p p if p ∈ P n − 1 p\in\mathcal{P}_{n-1} p ∈ P n − 1 , and (9.6.10) is trivially satisfied. Otherwise, the error formula (9.1.8) implies
p ( x ) − p ^ n ( x ) = p ( n ) ( ξ ( x ) ) n ! Φ ( x ) , p(x) - \hat{p}_n(x) = \frac{p^{(n)}(\xi(x))}{n!} \Phi(x), p ( x ) − p ^ n ( x ) = n ! p ( n ) ( ξ ( x )) Φ ( x ) , where Φ is the error indicator function ∏ k ( t − t k ) . \prod_k (t-t_k). ∏ k ( t − t k ) . Trivially, the left-hand side is a polynomial in P 2 n − 1 \mathcal{P}_{2n-1} P 2 n − 1 of degree at least n n n , so the right-hand side must be too. Thus, we can write
p ( x ) − p ^ n ( x ) = Ψ ( x ) Φ ( x ) , p(x) - \hat{p}_n(x) = \Psi(x) \Phi(x), p ( x ) − p ^ n ( x ) = Ψ ( x ) Φ ( x ) , where Ψ ( x ) ∈ P n − 1 \Psi(x)\in \mathcal{P}_{n-1} Ψ ( x ) ∈ P n − 1 is unknown. The optimality requirement (9.6.10) becomes
0 = ∫ − 1 1 p ( x ) d x − Q n [ p ] = ∫ − 1 1 [ p ( x ) − p ^ n ( x ) ] d x = ∫ − 1 1 Ψ ( x ) Φ ( x ) d x . 0 = \int_{-1}^1 p(x)\,dx - Q_{n}[p] = \int_{-1}^1 \bigl[p(x) - \hat{p}_n(x)\bigr]\,dx = \int_{-1}^1 \Psi(x) \Phi(x) \, dx. 0 = ∫ − 1 1 p ( x ) d x − Q n [ p ] = ∫ − 1 1 [ p ( x ) − p ^ n ( x ) ] d x = ∫ − 1 1 Ψ ( x ) Φ ( x ) d x . Given that Ψ ( x ) ∈ P n − 1 \Psi(x)\in \mathcal{P}_{n-1} Ψ ( x ) ∈ P n − 1 , we can ensure that this condition is satisfied if
∫ − 1 1 q ( x ) Φ ( x ) d x = 0 \int_{-1}^1 q(x)\Phi(x) \,dx = 0 ∫ − 1 1 q ( x ) Φ ( x ) d x = 0 for all q ∈ P n − 1 q \in {\mathcal{P}}_{n-1} q ∈ P n − 1 . Hence satisfaction of (9.6.17) implies satisfaction of (9.6.10) . But by the orthogonality property of Legendre polynomials, satisfaction of (9.6.17) is guaranteed if Φ ( x ) = c P n ( x ) \Phi(x)=cP_n(x) Φ ( x ) = c P n ( x ) for a constant c c c . Thus Φ and P n P_n P n have the same roots.
From Theorem 9.4.3 we know that the roots of P n P_n P n are distinct and all within ( − 1 , 1 ) (-1,1) ( − 1 , 1 ) . (Indeed, it would be strange to have the integral of a function depend on some of its values outside the integration interval!) While there is no explicit formula for the roots, there are fast algorithms to compute them and the integration weights on demand. Function 9.6.2 uses one of the oldest methods, practical up to n = 100 n=100 n = 100 or so.
Gauss–Legendre integration
Gauss-Legendre integration1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
function [I,x] = glint(f,n)
% GLINT Gauss-Legendre numerical integration.
% Input:
% f integrand (function)
% n number of nodes (integer)
% Output:
% I estimate of integral(f,-1,1)
% x evaluation nodes of f (vector)
% Nodes and weights are found via a tridiagonal eigenvalue problem.
beta = 0.5./sqrt(1-(2*(1:n-1)).^(-2));
T = diag(beta,1) + diag(beta,-1);
[V,D] = eig(T);
x = diag(D); [x,idx] = sort(x); % nodes
c = 2*V(1,idx).^2; % weights
% Evaluate the integrand and compute the integral.
I = c*f(x); % vector inner product
Gauss–Legendre integration1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
def glint(f, n):
"""
glint(f, n)
Perform Gauss-Legendre integration for the function f on n nodes in (-1,1). Return
integral and a vector of the nodes used.
"""
# Nodes and weights are found via a tridiagonal eigenvalue problem.
beta = 0.5 / np.sqrt(1 - (2.0 * np.arange(1, n)) ** (-2))
T = np.diag(beta, 1) + np.diag(beta, -1)
ev, V = eig(T)
ev = np.real_if_close(ev)
p = np.argsort(ev)
x = ev[p] # nodes
c = 2 * V[0, p] ** 2 # weights
# Evaluate the integrand and compute the integral.
I = np.dot(c, f(x)) # vector inner product
return I, x
9.6.4 Convergence ¶ Both Clenshaw–Curtis and Gauss–Legendre integration are spectrally accurate. The Clenshaw–Curtis method on n + 1 n+1 n + 1 points has degree n n n , whereas the Gauss–Legendre method with n n n points has degree 2 n − 1 {2n-1} 2 n − 1 . For this reason, it is possible for Gauss–Legendre to converge at a rate that is “twice as fast,” i.e., with roughly the square of the error of Clenshaw–Curtis. But the full story is not simple.
Example 9.6.3 (Comparing spectral integration methods)
Example 9.6.3 First consider the integral
∫ − 1 1 1 1 + 4 x 2 d x = arctan ( 2 ) . \int_{-1}^1 \frac{1}{1+4x^2} \, dx = \arctan(2). ∫ − 1 1 1 + 4 x 2 1 d x = arctan ( 2 ) . f(x)= 1 / (1 + 4x^2);
exact = atan(2);
We compare the two spectral integration methods for a range of n n n values.
using Plots
n = 8:4:96
err = zeros(length(n), 2)
for (k, n) in enumerate(n)
err[k, 1] = abs(exact - FNC.ccint(f, n)[1])
err[k, 2] = abs(exact - FNC.glint(f, n)[1])
end
err[iszero.(err)] .= NaN # remove from log-scale plot
plot(n, err, m=:o, label=["CC" "GL"],
xaxis=("number of nodes"), yaxis=(:log10, "error", [1e-16, 1]),
title="Spectral integration")
(The missing dots are where the error is exactly zero.) Gauss–Legendre does converge faster here, but at something less than twice the rate.
Now we try a more sharply peaked integrand:
∫ − 1 1 1 1 + 16 x 2 d x = 1 2 arctan ( 4 ) . \int_{-1}^1 \frac{1}{1+16x^2} \, dx = \frac{1}{2}\arctan(4). ∫ − 1 1 1 + 16 x 2 1 d x = 2 1 arctan ( 4 ) . f(x) = 1 / (1 + 16x^2);
exact = atan(4) / 2;
n = 8:4:96
err = zeros(length(n), 2)
for (k,n) in enumerate(n)
err[k, 1] = abs(exact - FNC.ccint(f, n)[1])
err[k, 2] = abs(exact - FNC.glint(f, n)[1])
end
err[iszero.(err)] .= NaN # remove from log-scale plot
plot(n, err, m=:o, label=["CC" "GL"],
xaxis=("number of nodes"), yaxis=(:log10, "error", [1e-16, 1]),
title="Spectral integration")
The two are very close until about n = 40 n=40 n = 40 , when the Clenshaw–Curtis method slows down.
Now let’s compare the spectral performance to that of our earlier adaptive method in intadapt
. We will specify varying error tolerances and record the error as well as the total number of evaluations of f f f .
tol = 10 .^(-2.0:-2:-14)
n = zeros(size(tol))
errAdapt = zeros(size(tol))
for (k, tol) in enumerate(tol)
Q, t = FNC.intadapt(f, -1, 1, tol)
errAdapt[k] = abs(exact - Q)
n[k] = length(t)
end
errAdapt[iszero.(errAdapt)] .= NaN
plot!(n, errAdapt, m=:o, label="intadapt")
plot!(n, n.^(-4), l=:dash, label="4th order",
xaxis=(:log10), title="Spectral vs 4th order" )
At the core of intadapt
is a fourth-order formula, and the results track that rate closely. For all but the most relaxed error tolerances, both spectral methods are far more efficient than the low-order counterpart. For other integrands, particularly those that vary nonuniformly across the interval, the adaptive method might be more competitive.
Example 9.6.3 First consider the integral
∫ − 1 1 1 1 + 4 x 2 d x = arctan ( 2 ) . \int_{-1}^1 \frac{1}{1+4x^2} \, dx = \arctan(2). ∫ − 1 1 1 + 4 x 2 1 d x = arctan ( 2 ) . f = @(x) 1 ./ (1 + 4*x.^2);
exact = atan(2);
We compare the two spectral integration methods for a range of n n n values.
n = (8:4:96)';
errCC = zeros(size(n));
errGL = zeros(size(n));
for k = 1:length(n)
errCC(k) = exact - ccint(f, n(k));
errGL(k) = exact - glint(f, n(k));
end
clf, semilogy(n, abs([errCC errGL]), 'o-')
xlabel('number of nodes'), ylabel('error')
title('Spectral integration')
legend('Clenshaw–Curtis', 'Gauss–Legendre')
(The missing points are where the error is exactly zero.) Gauss–Legendre does converge faster here, but at something less than twice the rate.
Now we try a more sharply peaked integrand:
∫ − 1 1 1 1 + 16 x 2 d x = 1 2 arctan ( 4 ) . \int_{-1}^1 \frac{1}{1+16x^2} \, dx = \frac{1}{2}\arctan(4). ∫ − 1 1 1 + 16 x 2 1 d x = 2 1 arctan ( 4 ) . f = @(x) 1 ./ (1 + 16*x.^2);
exact = atan(4) / 2;
n = (8:4:96)';
errCC = zeros(size(n));
errGL = zeros(size(n));
for k = 1:length(n)
errCC(k) = exact - ccint(f, n(k));
errGL(k) = exact - glint(f, n(k));
end
clf, semilogy(n, abs([errCC errGL]), 'o-')
xlabel('number of nodes'), ylabel('error')
title('Spectral integration')
legend('Clenshaw–Curtis', 'Gauss–Legendre')
The two are very close until about n = 40 n=40 n = 40 , when the Clenshaw–Curtis method slows down.
Now let’s compare the spectral performance to that of our earlier adaptive method in intadapt
. We will specify varying error tolerances and record the error as well as the total number of evaluations of f f f .
tol = 10 .^ (-2:-2:-14)';
n = zeros(size(tol));
errAdapt = zeros(size(tol));
for k = 1:length(n)
[Q, t] = intadapt(f, -1, 1, tol(k));
errAdapt(k) = exact - Q;
n(k) = length(t);
end
hold on; semilogy(n, abs(errAdapt), 'o-')
plot(n, n.^(-4), 'k--') % 4th order error
set(gca, 'xscale', 'log')
legend('ccint', 'glint', 'intadapt', '4th order')
title(('Spectral vs 4th order'));
At the core of intadapt
is a fourth-order formula, and the results track that rate closely. For all but the most relaxed error tolerances, both spectral methods are far more efficient than the low-order counterpart. For other integrands, particularly those that vary nonuniformly across the interval, the adaptive method might be more competitive.
Example 9.6.3 First consider the integral
∫ − 1 1 1 1 + 4 x 2 d x = arctan ( 2 ) . \int_{-1}^1 \frac{1}{1+4x^2} \, dx = \arctan(2). ∫ − 1 1 1 + 4 x 2 1 d x = arctan ( 2 ) . f = lambda x: 1 / (1 + 4 * x**2)
exact = arctan(2)
We compare the two spectral integration methods for a range of n n n values.
N = range(8, 100, 4)
errCC = zeros(len(N))
errGL = zeros(len(N))
for k, n in enumerate(N):
errCC[k] = exact - FNC.ccint(f, n)[0]
errGL[k] = exact - FNC.glint(f, n)[0]
semilogy(N, abs(errCC), "-o", label="Clenshaw–Curtis")
semilogy(N, abs(errGL), "-o", label="Gauss–Legendre")
xlabel("number of nodes"), ylabel("error"), ylim(1e-16, 0.01)
legend(), title("Spectral integration");
(The missing dots are where the error is exactly zero.) Gauss–Legendre does converge faster here, but at something less than twice the rate.
Now we try a more sharply peaked integrand:
∫ − 1 1 1 1 + 16 x 2 d x = 1 2 arctan ( 4 ) . \int_{-1}^1 \frac{1}{1+16x^2} \, dx = \frac{1}{2}\arctan(4). ∫ − 1 1 1 + 16 x 2 1 d x = 2 1 arctan ( 4 ) . f = lambda x: 1 / (1 + 16 * x**2)
exact = atan(4) / 2
N = range(8, 100, 4)
errCC = zeros(len(N))
errGL = zeros(len(N))
for k, n in enumerate(N):
errCC[k] = exact - FNC.ccint(f, n)[0]
errGL[k] = exact - FNC.glint(f, n)[0]
semilogy(N, abs(errCC), "-o", label="Clenshaw–Curtis")
semilogy(N, abs(errGL), "-o", label="Gauss–Legendre")
xlabel("number of nodes"), ylabel("error"), ylim(1e-16, 0.1)
legend(), title("Spectral integration");
The two are very close until about n = 40 n=40 n = 40 , when the Clenshaw–Curtis method slows down.
Now let’s compare the spectral performance to that of our earlier adaptive method in intadapt
. We will specify varying error tolerances and record the error as well as the total number of evaluations of f f f .
loglog(N, abs(errCC), "-o", label="ccint")
loglog(N, abs(errGL), "-o", label="glint")
tol_ = 1 / 10 ** arange(2, 15)
n = zeros(tol_.size)
errAdapt = zeros(tol_.size)
for k, tol in enumerate(tol_):
Q, t = FNC.intadapt(f, -1, 1, tol)
errAdapt[k] = exact - Q
n[k] = t.size
loglog(n, abs(errAdapt), "-o", label="intadapt")
loglog(n, 1 / (n**4), "--", label="4th order")
xlabel("number of nodes"), ylabel("error"), ylim(1e-16, 1)
legend(), title("Spectral vs 4th order");
At the core of intadapt
is a fourth-order formula, and the results track that rate closely. For all but the most relaxed error tolerances, both spectral methods are far more efficient than the low-order counterpart. For other integrands, particularly those that vary nonuniformly across the interval, the adaptive method might be more competitive.
The difference in convergence between Clenshaw–Curtis and Gauss–Legendre is dwarfed by the difference between spectral and algebraic convergence. It is possible, though, to encounter integrands for which adaptivity is critical. Choosing a method is highly problem-dependent, but a rule of thumb is that for large error tolerances, an adaptive low-order method is likely to be a good choice, while for high accuracy, the spectral methods often dominate.
9.6.5 Exercises ¶ ✍ Suppose f f f is periodic on [ − 1 , 1 ] [-1,1] [ − 1 , 1 ] . Show that the result of applying the trapezoid formula on 2 n + 1 2n+1 2 n + 1 points is identical to (9.6.2) .
✍ For each integral, use Gauss–Legendre integration with n = 2 n=2 n = 2 to write out the terms w 1 f ( t 1 ) w_1f(t_1) w 1 f ( t 1 ) and w 2 f ( t 2 ) w_2f(t_2) w 2 f ( t 2 ) explicitly.
(a) ∫ − 1 1 e − x d x = 2 sinh ( 1 ) \displaystyle\int_{-1}^1 e^{-x}\, dx = 2 \sinh(1) \qquad ∫ − 1 1 e − x d x = 2 sinh ( 1 )
(b) ∫ − 1 1 e − x 2 \displaystyle\int_{-1}^1 e^{-x^2} \qquad ∫ − 1 1 e − x 2
(c) ∫ − 1 1 ( 2 + x ) − 1 \displaystyle\int_{-1}^1 (2+x)^{-1} ∫ − 1 1 ( 2 + x ) − 1
⌨ For each integral, compute approximations using Function 9.6.1 and Function 9.6.2 with n = 4 , 6 , 8 , … , 40 n=4,6,8,\ldots,40 n = 4 , 6 , 8 , … , 40 . Plot the errors of both methods together as functions of n n n on a semi-log scale.
(a) ∫ − 1 1 e − 4 x d x = sinh ( 4 ) / 2 \displaystyle\int_{-1}^1 e^{-4x}\, dx = \sinh(4)/2 ∫ − 1 1 e − 4 x d x = sinh ( 4 ) /2
(b) ∫ − 1 1 e − 9 x 2 = π erf ( 3 ) / 3 \displaystyle\int_{-1}^1 e^{-9x^2} = \sqrt{\pi}\, \operatorname{erf}(3)/3 ∫ − 1 1 e − 9 x 2 = π erf ( 3 ) /3
(c) ∫ − 1 1 sech ( x ) d x = 2 tan − 1 [ sinh ( 1 ) ] \displaystyle\int_{-1}^1 \operatorname{sech}(x) \, dx = 2 \tan^{-1} [ \sinh (1) ] ∫ − 1 1 sech ( x ) d x = 2 tan − 1 [ sinh ( 1 )]
(d) ∫ − 1 1 1 1 + 9 x 2 d x = 2 3 tan − 1 ( 3 ) \displaystyle\int_{-1}^1 \frac{1}{1+9x^2}\, dx = \frac{2}{3} \tan^{-1}(3) ∫ − 1 1 1 + 9 x 2 1 d x = 3 2 tan − 1 ( 3 )
(a) ✍ (See also Exercise 9.3.5 .) Using the change of variable
z = ϕ ( x ) = a + ( b − a ) ( x + 1 ) 2 , z = \phi(x) = a + (b-a)\frac{(x+1)}{2}, z = ϕ ( x ) = a + ( b − a ) 2 ( x + 1 ) , show that
∫ a b f ( z ) d z = b − a 2 ∫ − 1 1 f ( ϕ ( x ) ) d x . \int_{a}^b f(z) \, d z= \frac{b-a}{2} \int_{-1}^{1} f( \phi(x) ) \, d x . ∫ a b f ( z ) d z = 2 b − a ∫ − 1 1 f ( ϕ ( x )) d x . (b) ⌨ Rewrite Function 9.6.1 and Function 9.6.2 to accept additional inputs for a a a and b b b and compute integrals over [ a , b ] [a,b] [ a , b ] .
(c) ⌨ Repeat the steps of Exercise 2 for the integral
∫ π / 2 π x 2 sin 8 x d x = − 3 π 2 32 . \int_{\pi/2}^{\pi} x^2 \sin 8x \, d x = -\frac{3 \pi^2}{32}. ∫ π /2 π x 2 sin 8 x d x = − 32 3 π 2 . ⌨ A particle moves in a circular path with angular velocity given by ω ( θ ) = sin ( exp ( sin θ ) ) \omega(\theta)=\sin(\exp(\sin \theta)) ω ( θ ) = sin ( exp ( sin θ )) . The time it takes to complete one full orbit is
T = ∫ 0 T d t = ∫ 0 2 π d θ d θ / d t = ∫ 0 2 π d θ ω ( θ ) . T = \int_0^T dt = \int_{0}^{2\pi} \frac{d\theta}{d\theta / dt } = \int_{0}^{2\pi} \frac{d\theta}{\omega(\theta) }. T = ∫ 0 T d t = ∫ 0 2 π d θ / d t d θ = ∫ 0 2 π ω ( θ ) d θ . Use Function 5.6.1 to find the period of the motion. Show results for different values of n n n to establish convergence to at least 12 digits.
✍ Prove the claim about linearity of the Gauss–Legendre integration formula alluded to in the derivation of Theorem 9.6.1 . Namely, show that condition (9.6.10) is true if and only if
∫ − 1 1 x j d x = ∑ k = 1 n w k x k j \int_{-1}^1 x^j\,dx = \sum_{k=1}^n w_k x_k^j ∫ − 1 1 x j d x = k = 1 ∑ n w k x k j for all j = 0 , … , 2 n − 1 j=0,\ldots,2n-1 j = 0 , … , 2 n − 1 .