This has the following generic form, which is common to other numerical integration (quadrature) methods:
$$I = \int_a^bf(x)dx \approx \sum_{i=0}^{n-1}w_if(x_i).$$That is, the integral is approximated as some sum over the function at points $x_i$, multiplied by some coefficients $w_i$. For the Trapezoid rule, $w_i=\Delta x$ for the interior points and and $w=\Delta x/2$ for the edge points.
Gauss Quadrature selects the $x_i$ and the $w_i$ so that the integrals of polynomials up to degree $2n-1$ is exact.
This can be extended to higher degree using higher $n$.
To the extent that our function is well represented by a degree $2n-1$ polynomial, the following approximation is accurate:
import numpy as np
import matplotlib.pyplot as plt
from scipy.integrate import quad
%matplotlib inline
a = np.random.rand(4)
Igq = np.polyval(a, -1.0/np.sqrt(3)) + np.polyval(a, 1.0/np.sqrt(3))
Ipy = quad(lambda z: np.polyval(a, z), -1, 1)[0]
print('I Gauss Quadrature: ', Igq)
print('I Python quad func: ', Ipy)
I Gauss Quadrature: 1.7764078505045668 I Python quad func: 1.776407850504567
from numpy.polynomial.legendre import leggauss
n = 2
z, w = leggauss(n)
print('abscissas: z_i', z)
print('weights : w_i', w)
abscissas: z_i [-0.57735027 0.57735027] weights : w_i [1. 1.]
def f(x):
return 4+(x-5)**3
a = 3
b = 7
m = (b-a)/2
c = (b+a)/2
I_fxab = quad(f, a, b)[0]
I_fz11 = m*quad(lambda z: f(m*z+c), -1, 1)[0]
print(f"Compare integral over a<x<b to integral over -1<z<1 using quad function: {I_fxab}, {I_fz11}")
#--------------
n = 2
z, w = leggauss(n)
I_GQ = m*np.sum(w*f(m*z+c))
print("Compare to Gauss Quadrature: ", I_GQ)
Compare integral over a<x<b to integral over -1<z<1 using quad function: 16.0, 16.0 Compare to Gauss Quadrature: 16.0
def f(x): # define the function
#return np.exp(x)
return np.sin(x)
#return x**8
#return x**(5/3)
#return np.sqrt(x)
#return x**(-1/3)
a = 3
b = 7
m = (b-a)/2
c = (b+a)/2
#Iexact = np.exp(b) - np.exp(a) # exact integral
Iexact = -np.cos(b) + np.cos(a) # exact integral
#Iexact = 1/9*b**9 - 1/9*a**9 # exact integral
#Iexact = 3/8*b**(8/3)- 3/8*a**(8/3) # exact integral
#Iexact = 2/3*b**(3/2)- 2/3*a**(3/2) # exact integral
#Iexact = 3/2*b**(2/3) - 3/2*a**(2/3) # exact integral
#Iexact = quad(f,a,b)[0] # "exact" integral: quad function
n = np.arange(2,18) # list of number of quadrature points to try
ncases = len(n) # number of cases
Ierr = np.empty(ncases) # array of relative errors
for i,ni in enumerate(n):
z, w = leggauss(ni)
Ierr[i] = np.abs((m*np.sum(f(m*z+c)*w) - Iexact)/Iexact)
#-------------- Compare to trapezoid method with the same number of points:
def Itrap(f, a,b,n):
x = np.linspace(a,b,n)
dx = x[1] - x[0]
return dx*np.sum(f(x)) - dx/2*(f(x[0])+f(x[-1]))
Ierr_trap = np.empty(ncases)
for i, ni in enumerate(n):
Ierr_trap[i] = Itrap(f, a, b, ni)
Ierr_trap = np.abs( (Ierr_trap - Iexact)/Iexact )
#------------- plot
plt.rc('font', size=14)
plt.plot(n, np.log10(Ierr_trap), 'o-');
plt.plot(n, np.log10(Ierr), 'o-');
plt.xlabel("number of points");
plt.ylabel(r'$\log_{10}$ relative error');
plt.legend(['Trapezoid', 'Gauss quadrature'], frameon=False);
$$-\infty<x<\infty$$
Consider a particle size distribution $n(m)$, where $m$ is mass of a particle, and $n$ is number of particles per unit volume per unit mass.
The $k^{th}$ moment of the distribution is given by
$M_k = \int n(m)m^k dm$.
(Notice the resemblance to a weighted Gauss quadrature if $n(m)$ is our weight function.
We care most about the first two moments $M_0$ and $M_1$.
The more moments that are solved, the more accurate the results, especially $M_0$ and $M_1$.
Suppose we know how the size distribution $n(m)$ evolves. That is, we know
$$\frac{dn(m)}{dt} = S(m, n(m)).$$The source term $S$ accounts for collisions, chemical growth, etc. (If two particles collide and stick, then $n(m)$ shifts to have more particles at larger sizes.)
We can convert this into an equation for $dM_k/dt$ by multiplying by $m^k$ and integrating over all sizes. This gives
$$\frac{dM_k}{dt} = \int m^kS(m, n(m))dm.$$So, we evolve the moments in time according to this equation.
We treat the integral with quadrature, using $n(m)$ as our weight function.
Now, we need the $m_i$ and the $w_i$.
Using the definition of the moments we have
\begin{align} M_0 &= \int m^0n(m)dm = m_0^0w_0 + m_1^0w_1, \\ M_1 &= \int m^1n(m)dm = m_0^1w_0 + m_1^1w_1, \\ M_2 &= \int m^2n(m)dm = m_0^2w_0 + m_1^2w_1, \\ M_3 &= \int m^3n(m)dm = m_0^3w_0 + m_1^3w_1. \end{align}Solve this system for $m_0$, $m_1$, $w_0$, $w_1$, then we can compute the integrals in the $dM_k/dt$ equations and the system is closed.
Note how we evaluate the moment source terms in terms of $n(m)$ without ever having to know $n(m)$. This works because we are dealing with integrals using a weighted Gauss quadrature where we $do$ know the moments. Those moments give use the weights and abscissas for the quadrature, allowing us to do the integrations.
This is an extremely accurate and powerful method.