Conjugate Gradient Methods

Quadratic Forms and Linear Systems

Many important quantities arising in engineering are mathematically modelled as Quadratic Forms. Output noise power in Signal Processing, Kinetic and Potential Energy functions in Mechanical Engineering, Covariance Models of Random Variable Linear Dependence and Principal Component Analysis in Statistics, etc. are some of the frequently occurring applications of Quadratic Forms, illustrating their ubiquity and importance.

A Quadratic Form \(f(x)\) on \(\mathbb{R}^n\) is a function \(f : \mathbb{R}^n \rightarrow \mathbb{R}\) such that:

\begin{align*} f(\mathbf{x}) = \frac{1}{2} \mathbf{x}^T H \mathbf{x} + \mathbf{c}^T x \\ \end{align*}
\begin{align*} \mathbf{c}, \mathbf{x} \in \mathbb{R}^n, H \in \mathbb{R}^{ n \times n} \text{ for simplicity} \\ \end{align*}
\begin{align*} H = H^T, \text{ and } \mathbf{x}^T H \mathbf{x} > 0 \: \forall \: \mathbf{x} \neq 0, \mathbf{x} \in \mathbb{R}^{\, n} \\ \end{align*}

The graph of a positive-definite convex quadratic form is a hyperparaboloid in \(\mathbb{R}^{n + 1}\) dimensions.

Frequently, in many applications, we are interested in minimizing the quadratic form \(f(x)\), for example, minimising the energy function of a mechanical system so that it is in a stable state.

Suppose that, we have at our hands, a quadratic form obtained by mathematically modelling some quantity of interest and we wish to minimize it. At the minimum point \(\mathbf{x}^*\) (henceforth referred to as the minimizer), the gradient \(\nabla : \mathbb{R}^n \rightarrow \mathbb{R}\) of a function \(f(x)\) is always zero by first order optimality condition. Hence,

\begin{align*} \nabla f(\mathbf{x}) = \frac{1}{2} \left[ H + H^T \right] \mathbf{x} + \mathbf{c} \\ \end{align*}
\begin{align*} \nabla f(\mathbf{x}) = H \mathbf{x} + \mathbf{c} \text{ since } H = H^T \\ \end{align*}

At \(\mathbf{x}^*\), \(\nabla f(\mathbf{x}^*) = 0 \implies H \mathbf{x}^* + \mathbf{c} = 0 \text{ or } H \mathbf{x}^* = -\mathbf{c}\).

Thus, finding the minimum of the quadratic form \(f(\mathbf{x})\) is equivalent to solving the linear system \(H \mathbf{x}^* = -\mathbf{c}\) for the unknown \(\mathbf{x}^*\).

Again, suppose that we have at our hands a one-dimensional two-point boundary value problem (BVP), for example, the governing equations of flow of heat in a thin conducting rod with a source (\(t\) is the continuous 1D spatial domain, \(x\) is the temperature, \(g(t)\) is the heat source):

\begin{align*} \frac{d^2 x}{dt^2} = g(t) \\ \end{align*}

If we discretise the domain and convert the above differential equation to a finite difference equation, we obtain a linear system \(H \mathbf{x} = - \mathbf{c}\), where the temperatures \(\mathbf{x}\) can be a very high dimensional vector due to the underlying problem being continuous. The \(H\) represents the second-order derivative operator \(\frac{d^2 x}{dt^2}\) and \(- \mathbf{c}\) the heat source in the discretised domain \(\mathbf{R}^n\).

Since the BVP is continuous, the discretisation \(x\) is made high-dimensional to capture its fidelity; therefore the matrix \(H \in \mathbb{R}^{n \times n}\) is very large. More importantly, due to the structure and regularity of the 2nd order finite difference operator, we can make the following qualifiers: the matrix \(H\) is sparse, symmetric and positive definite. Solving such a large linear system with a direct matrix factorisation method is prohibitively expensive (of the order of \(\mathcal{O}(\frac{2}{3} n^3)\)). Hence, a better alternative is to solve the system iteratively and reach a solution in \(r << n\) steps that is close enough to the actual solution, which will cost us much less than a direct method.

One idea to solve the linear system \(H \mathbf{x} = - \mathbf{c}\) is to use a stationary method, such as Jacobi or Gauss-Seidel. However, there is no guarantee of convergence, even for positive definite matrices.

A better approach is to utilize the observation made earlier: convert the linear system to its dual quadratic form and apply an iterative optimisation process on the objective function!

\begin{align*} H \mathbf{x} = - \mathbf{c} \\ \end{align*}
\begin{align*} \Leftrightarrow \underset{\mathbf{x} \in \mathbb{R}^n}{\mathrm{min}} \: f(\mathbf{x}) = \frac{1}{2} \mathbf{x}^T H \mathbf{x} + \mathbf{c}^T x \\ \end{align*}

At the minimum of \(f(\mathbf{x})\), we would have found the solution to our original linear system.

We can summarise this section by making the following statement [See PCG Appendix C1 for proof]:

Line Search Techniques

Now that we have our objective function, namely the quadratic form \(f : \mathbb{R}^n \rightarrow \mathbb{R}\), our goal is to minimise it. We formalise our unconstrained optimisation problem \(\text{(P)}\) as:

\begin{align*} \text{(P)} \quad \underset{\mathbf{x} \in \mathbb{R}^n}{\mathrm{min}} \: f(\mathbf{x}) = \frac{1}{2} \mathbf{x}^T H \mathbf{x} + \mathbf{c}^T x \\ \end{align*}
Before we begin, let us introduce some definitions:
  1. Error at the i-th iterate: \(\mathbf{e}_i = \mathbf{x}_i - \mathbf{x}^*\)
  2. Residual at the i-th iterate: \(\mathbf{r}_i = H \mathbf{x}_i + \mathbf{c}\), from linear system perspective.
  3. Gradient at the i-th iterate: \(\nabla f(\mathbf{x}_i) = H \mathbf{x}_i + \mathbf{c}\), from objective function perspective.
We notice that:
  1. \(\mathbf{r}_i = H \mathbf{x}_i + \mathbf{c} = H \mathbf{x}_i - H \mathbf{x}^* = H (\mathbf{x}_i - \mathbf{x}^*) = H \mathbf{e}_i\), that is, the residual is simply the error mapped from the domain of \(H\) to the range of \(H\).
  2. The Residual of the Linear System is equal to the Gradient of the Objective Function. Throughout the article, whenever we mention residual, we also mention gradient to reinforce this fact.
  3. Since \(\mathbf{x}^*\) is unknown, \(\mathbf{e}_i\) is unknown at every step, but the residuals \(\mathbf{r}_i\) are always known. So whenever we want to use the error, we can simply work with the residual in the rangespace of \(H\).

Now comes the core philosophy of Line Search: given our objective function \(f(\mathbf{x})\), we start with an initial guess \(\mathbf{x}_0\), and iterate our way downhill on \(f(\mathbf{x})\) to reach \(\mathbf{x}^*\). At any i-th iterate, we are at the point \(\mathbf{x}_i\), and to travel to our next point \(\mathbf{x}_{i + 1}\), we must choose a direction of descent \(\mathbf{p}_i\), and then move a step length \(\alpha_i\) the right amount so that along this direction, \(f(\mathbf{x}_{i + 1}) = \phi(\alpha_i)\) is minimum. Mathematically,

\begin{align*} \mathbf{x}_{i + 1} = \mathbf{x}_i + \alpha_i \mathbf{p}_i \\ \end{align*}
\begin{align*} \alpha_i = \underset{x_{i + 1}}{\mathrm{argmin}} \: f(\mathbf{x}_{i + 1}) = \underset{\alpha_{i}}{\mathrm{argmin}} \: f(\mathbf{x}_i + \alpha_i \mathbf{p}_i) = \underset{\alpha_{i}}{\mathrm{argmin}} \: \phi(\alpha_i) \\ \end{align*}

One approach is to use the Steepest Descent (SD) technique to compute and move along a direction \(\mathbf{p}_i\) towards vanishing gradient:

\begin{align*} \mathbf{p}_i = - \nabla f(\mathbf{x}_i) = - \mathbf{r}_i \\ \end{align*}

SD moves along (or opposite to be precise) the residual/gradient vector at each iterate. One fact we state without proof [see PCG pg. 6; can be derived from 1D calculus] is that the residual/gradient at the current iterate i is always orthogonal to the previous residual/gradient at i - 1.

As can be seen from the above picture, this is unfortunate, because we repeat directions in multiple steps a lot, and the number of steps needed to converge to the minimum point \(\mathbf{x}^*\) is extraordinarily large. This could have been avoided if, after taking a new direction each time, we had travelled just the right amount of length in that direction so that in our future march downhill, we never need to step in that direction again.

This raises an important question: If we picked a direction and took the RIGHT step length along it to avoid repeating it, in how many steps will be reach the minimizer \(\mathbf{x}^*\)?

The answer is: n steps! (the dimension of the vectorspace). To see why it is so, we need to wait for some more explanations.

Let us the explore the above idea a bit further in the following sections.

Coordinate Descent Method

To demonstrate the idea of n-directions, let us solve two problems and analyse their solutions. As a first attempt, we will pick the coordinate axes (i.e. the canonical basis) as the directions of descent.

Problem 1

\begin{align*} \underset{x \in \mathbb{R}^2}{\mathrm{min}} \: f(\mathbf{x}) = 4 x_1^2 + x_2^2 = \frac{1}{2} \begin{bmatrix} x_1 \\ x_2 \end{bmatrix}^T \begin{bmatrix} 8 & 0\\ 0 & 2 \end{bmatrix} \begin{bmatrix} x_1 \\ x_2 \end{bmatrix} + \begin{bmatrix} 0 \\ 0 \end{bmatrix}^T \begin{bmatrix} x_1 \\ x_2 \end{bmatrix} \\ \end{align*}

The minimum point is at \(\mathbf{x}^* = \begin{bmatrix} 0 \\ 0 \end{bmatrix}\).

Let us select our initial guess as \(\mathbf{x}_0 = \begin{bmatrix} -1 \\ -1 \end{bmatrix}\). Our first direction of motion is \(\mathbf{p}_0 = \mathbf{e}_1 = \begin{bmatrix} 1 \\ 0 \end{bmatrix}\). Therefore, \(\mathbf{x}_1 = \mathbf{x}_0 + \alpha_0 \mathbf{p}_0 = \begin{bmatrix} -1 \\ -1 \end{bmatrix} + \alpha_0 \begin{bmatrix} 1 \\ 0 \end{bmatrix} = \begin{bmatrix} -1 + \alpha_0 \\ -1 \end{bmatrix}\) and \(\phi(\alpha_0) = f(\mathbf{x}_1) = 4 (-1 + \alpha_0)^2 + (-1)^2\). Taking the derivative of \(\phi(\alpha_0)\) with respect to \(\alpha_0\), setting it to zero and solving the equation, we obtain \(\alpha_0 = 1\). Therefore, \(\mathbf{x}_1 = \begin{bmatrix} 0 \\ -1 \end{bmatrix}\). Repeating this process for the next direction \(\mathbf{p}_1 = \mathbf{e}_2 = \begin{bmatrix} 0 \\ 1 \end{bmatrix}\), we obtain \(\mathbf{x}_2 = \begin{bmatrix} 0 \\ 0 \end{bmatrix}\), which is indeed our minimum \(\mathbf{x}^*\). We reached our destination in two steps! (= dimension of our decision space \(\mathbb{R}^2\))

The contours of the function and our iteration path are as displayed below:

Problem 2

\begin{align*} \underset{x \in \mathbb{R}^2}{\mathrm{min}} \: f(\mathbf{x}) = 4 x_1^2 + x_2^2 - 2 x_1 x_2 = \frac{1}{2} \begin{bmatrix} x_1 \\ x_2 \end{bmatrix}^T \begin{bmatrix} 8 & -2 \\ -2 & 2 \end{bmatrix} \begin{bmatrix} x_1 \\ x_2 \end{bmatrix} + \begin{bmatrix} 0 \\ 0 \end{bmatrix}^T \begin{bmatrix} x_1 \\ x_2 \end{bmatrix} \\ \end{align*}

The minimum point is again at \(\mathbf{x}^* = \begin{bmatrix} 0 \\ 0 \end{bmatrix}\).

We repeat the same process as outlined in Problem 1, starting from \(\mathbf{x}_0 = \begin{bmatrix} -1 \\ -1 \end{bmatrix}\), and picking \(\mathbf{e}_1\) and then \(\mathbf{e}_2\) as search directions. However, this time, we obtain the iterates as \(\mathbf{x}_1 = \begin{bmatrix} \frac{-1}{4} \\ -1 \end{bmatrix}\), and \(\mathbf{x}_2 = \begin{bmatrix} \frac{-1}{4} \\ \frac{-1}{4} \end{bmatrix}\). Notice that, we haven't reached our minimizer in two steps.

Let us spend sometime looking over what we have done so far and why our coordinate axes approach failed in the second problem. Notice that, in Problem 1, the Hessian Matrix (ie the second derivative of a multivariate scalar function) of the Quadratic Form is diagonal, and hence our coordinate axes are aligned with its eigenvectors. Not only that, we know that for a symmetric matrix, the eigenvectors are always orthogonal, that is, \(\mathbf{v}_i^T \mathbf{v}_j = \left\{ \begin{array}{ll} \text{ non-zero } & \mbox{if } i = j \\ 0 & \mbox{if } i \neq j \end{array} \right. \: \forall \: i, j \: \in \: [1 \dots n]\).

We can write the minimizer as \(\mathbf{x}^* = \mathbf{x}_0 + \alpha_0 \mathbf{p}_0 + \alpha_1 \mathbf{p}_1 = \mathbf{x}_0 + \alpha_0 \mathbf{v}_1 + \alpha_1 \mathbf{v}_2\). Substitute in \(\mathbf{x}^*\) in the objective function and we will get:

\begin{align*} \underset{x \in \mathbb{R}^2}{\mathrm{min}} \: f(\mathbf{x}^*) = \frac{1}{2} (\mathbf{x}_0 + \alpha_0 \mathbf{v}_1 + \alpha_1 \mathbf{v}_2)^T \begin{bmatrix} 8 & 0\\ 0 & 2 \end{bmatrix} (\mathbf{x}_0 + \alpha_0 \mathbf{v}_1 + \alpha_1 \mathbf{v}_2) \\ \end{align*}
\begin{align*} \implies \underset{x \in \mathbb{R}^2}{\mathrm{min}} \: f(\mathbf{x}^*) = \frac{1}{2} (\mathbf{x}_0 + \alpha_0 \mathbf{v}_1 + \alpha_1 \mathbf{v}_2)^T H (\mathbf{x}_0 + \alpha_0 \mathbf{v}_1 + \alpha_1 \mathbf{v}_2) \\ \end{align*}
\begin{align*} \implies \underset{x \in \mathbb{R}^2}{\mathrm{min}} \: f(\mathbf{x}^*) = \frac{1}{2} (\mathbf{x}_0 + \alpha_0 \mathbf{v}_1 + \alpha_1 \mathbf{v}_2)^T (H\mathbf{x}_0 + \alpha_0 H \mathbf{v}_1 + \alpha_1 H \mathbf{v}_2) \\ \end{align*}
\begin{align*} \implies \underset{x \in \mathbb{R}^2}{\mathrm{min}} \: f(\mathbf{x}^*) = \frac{1}{2} \mathbf{x}_0^T (H\mathbf{x}_0 + \alpha_0 \lambda_1 \mathbf{v}_1 + \alpha_1 \lambda_2 \mathbf{v}_2) + \frac{1}{2} (\alpha_0 \mathbf{v}_1 + \alpha_1 \mathbf{v}_2)^T H\mathbf{x}_0 + \\ \frac{1}{2} (\alpha_0 \mathbf{v}_1 + \alpha_1 \mathbf{v}_2)^T (\alpha_0 H \mathbf{v}_1 + \alpha_1 H\mathbf{v}_2) \\ \end{align*}
\begin{align*} \implies \underset{x \in \mathbb{R}^2}{\mathrm{min}} \: f(\mathbf{x}^*) = \frac{1}{2} \mathbf{x}_0^T H\mathbf{x}_0 + \frac{1}{2} \alpha_0 \lambda_1 \mathbf{x}_0^T \mathbf{v}_1 + \frac{1}{2} \alpha_1 \lambda_2 \mathbf{x}_0^T \mathbf{v}_2 + \frac{1}{2} \alpha_0 \mathbf{v}_1^T H \mathbf{x}_0 + \frac{1}{2} \alpha_1 \mathbf{v}_2^T H \mathbf{x}_0 + \\ \left[ \frac{1}{2} \alpha_0^2 \mathbf{v}_1^T H \mathbf{v}_1 + \frac{1}{2} \alpha_0 \alpha_1 \mathbf{v}_1^T H \mathbf{v}_2 + \frac{1}{2} \alpha_1 \alpha_0 \mathbf{v}_2^T H \mathbf{v}_1 + \frac{1}{2} \alpha_1^2 \mathbf{v}_2^T H \mathbf{v}_2 \right] \\ \end{align*}

Let us pause for a moment here. The above equation looks very complicated, by we have to remember that \(\mathbf{v}_1\) and \(\mathbf{v}_2\) are eigenvectors of \(H\). Hence, the terms containing \(\mathbf{v}_1^T H \mathbf{v}_2\) and \(\mathbf{v}_2^T H \mathbf{v}_1\) become, \(\mathbf{v}_1^T H \mathbf{v}_2 = \mathbf{v}_1^T \lambda_2 \mathbf{v}_2 = \lambda_2 \mathbf{v}_1^T \mathbf{v}_2 = 0\) and \(\mathbf{v}_2^T H \mathbf{v}_1 = \mathbf{v}_2^T \lambda_1 \mathbf{v}_1 = \lambda_1 \mathbf{v}_2^T \mathbf{v}_1 = 0\), due to the orthogonality of the eigenvectors. Hence, after some simplifications, we get,

\begin{align*} \implies \underset{x \in \mathbb{R}^2}{\mathrm{min}} \: f(\mathbf{x}^*) = \frac{1}{2} \mathbf{x}_0^T H\mathbf{x}_0 + \alpha_0 \lambda_1 \mathbf{x}_0^T \mathbf{v}_1 + \alpha_1 \lambda_2 \mathbf{x}_0^T \mathbf{v}_2 + \frac{1}{2} (\alpha_0^2 \lambda_1 \mathbf{v}_1^T \mathbf{v}_1 + \alpha_1^2 \lambda_2 \mathbf{v}_2^T \mathbf{v}_2) \\ \end{align*}

In the derivation above, the products \(\mathbf{v}_i^T H \mathbf{v}_j\) became 0 whenever \(i \neq j\), but survived otherwise if \(i = j\). There are no mixed terms containing both directions \(\mathbf{v}_1\) and \(\mathbf{v}_2\) together, only in isolation, and we know that we have reached the minimiser using this technique.

We will do the same exercise for Problem 2. However, since the coordinate axes are not aligned with the eigenvectors of \(H\) because it is not diagonal, we cannot do the \(\mathbf{p}_i = \mathbf{v}_{i + 1}\) substitutions here. Therefore, (skipping some steps),

\begin{align*} \underset{x \in \mathbb{R}^2}{\mathrm{min}} \: f(\mathbf{x}) = \frac{1}{2} (\mathbf{x}_0 + \alpha_0 \mathbf{p}_0 + \alpha_1 \mathbf{p}_1)^T \begin{bmatrix} 8 & -2\\ -2 & 2 \end{bmatrix} (\mathbf{x}_0 + \alpha_0 \mathbf{p}_0 + \alpha_1 \mathbf{p}_1) \\ \end{align*}
\begin{align*} \implies \underset{x \in \mathbb{R}^2}{\mathrm{min}} \: f(\mathbf{x}) = \frac{1}{2} \mathbf{x}_0^T H\mathbf{x}_0 + \frac{1}{2} \alpha_0 \mathbf{x}_0^T H \mathbf{p}_0 + \frac{1}{2} \alpha_1 \mathbf{x}_0^T H \mathbf{p}_1 + \frac{1}{2} \alpha_0 \mathbf{p}_0^T H \mathbf{x}_0 + \frac{1}{2} \alpha_1 \mathbf{p}_1^T H \mathbf{x}_0 + \\ \left[ \frac{1}{2} \alpha_0^2 \mathbf{p}_0^T H \mathbf{p}_0 + \frac{1}{2} \alpha_0 \alpha_1 \mathbf{p}_0^T H \mathbf{p}_1 + \frac{1}{2} \alpha_1 \alpha_0 \mathbf{p}_1^T H \mathbf{p}_0 + \frac{1}{2} \alpha_1^2 \mathbf{p}_1^T H \mathbf{p}_1 \right] \\ \end{align*}

Again, let us compare this expression with that of Problem 1. The terms outside brackets are exactly the same for both problems. So, should not we reach the minimiser in this case also? If we look inside the sum inside the brackets, it is observed that the cross terms \(\mathbf{p}_i^T H \mathbf{p}_j \: \forall \: i \neq j\) are still present. This is the only difference between the two problems, and we have witnessed that one led to the solution, the other did not.

This leads us to a good start. We can take the eigenvectors of the Hessian \(H\) (SPD matrices always have a full set of n eigenvectors) and move along those directions, and we are guaranteed to reach the minimiser in n steps. BUT! Calculating eigenvectors of a matrix is very expensive. Using direct matrix factorisation sounds much better than this approach!

We also cannot use the residuals as search directions, as that will take us back to the Steepest Descent, and moreover they will not be linearly independent due to repetitions.

So what we will do instead is, we will build a set of n linear independent set of directions \(\mathbf{p}_i\), and impose the desirable property observed earlier from the eigenvectors that \(\mathbf{p}_i^T H \mathbf{p}_j = \left\{ \begin{array}{ll} \text{ non-zero } & \mbox{if } i = j \\ 0 & \mbox{if } i \neq j \end{array} \right. \: \forall \: i, j \: \in \: [1 \dots n]\). What we hope to achieve is the same effect we observed in Problem 1: no mixed direction terms in the objective function at the minimiser, with guaranteed convergence in n-steps.

This is the precisely the definition of Conjugate Directions! The idea is very simple, instead of clueless wandering in search of our destination by only using local gradient information, we intelligently exploit the nature of our problem to fix all our search directions in one shot. After n searches, we are done and can go home.

Before we move ahead, let us spend a little more time on Problem 1. Observe that, when moving from iterate \(\mathbf{x}_0\) to \(\mathbf{x}_1\), the error \(\mathbf{e}_1 = \mathbf{x}_1 - \mathbf{x}^*\) is orthogonal to the the direction \(\mathbf{p}_0\). This is another desirable property we will use in n-direction approach. However, we don't know \(\mathbf{x}^*\), so we don't know \(\mathbf{e}_{i + 1}\), and as noted in the previous section, we will instead make the residual/gradient \(\mathbf{r}_{i + 1} = H \mathbf{e}_{i + 1}\) orthogonal to the the previous search directions \(\mathbf{p}_i\).

Descent along Conjugate Directions: A Powerful Line Search Method

Two vectors, \(\mathbf{p}_i\) and \(\mathbf{p}_j\) are said to be H-conjugate with respect to some symmetric, positive definite matrix \(H\), if they satisfy,

\begin{align*} \mathbf{p}_i^T H \mathbf{p}_j = \left\{ \begin{array}{ll} \text{ non-zero } & \mbox{if } i = j \\ 0 & \mbox{if } i \neq j \end{array} \right. \: \forall \: i, j \: \in \: [1 \dots n] \\ \end{align*}

From the discussion in the previous section, we outline our goals and issues for the rest of the article as follows:

  1. Find a set of n linearly independent set of H-conjugate directions \(\mathbf{p}_i\).
  2. Establish relationships between residuals/gradients, search directions and the vectorspace of our solution \(\mathbf{x}^*\).
  3. Use the above two rules to calculate the step lengths \(\alpha_i\) along \(\mathbf{p}_i\).

To repeat, we are solving the unconstrained optimisation problem:

\begin{align*} \text{(P)} \quad \underset{\mathbf{x} \in \mathbb{R}^n}{\mathrm{min}} \: f(\mathbf{x}) = \frac{1}{2} \mathbf{x}^T H \mathbf{x} + \mathbf{c}^T x \\ \end{align*}
\begin{align*} \mathbf{c}, \mathbf{x} \in \mathbb{R}^n, H \in \mathbb{R}^{ n \times n} \\ \end{align*}
\begin{align*} H = H^T, \text{ and } \mathbf{x}^T H \mathbf{x} > 0 \: \forall \: \mathbf{x} \neq 0, \mathbf{x} \in \mathbb{R}^{\, n} \\ \end{align*}

The minimiser of our problem \(\text{(P)}\) can be stated as:

\begin{align*} \mathbf{x}^* = \mathbf{x}_0 + \sum_{i = 0}^{n - 1} \alpha_i \mathbf{p}_i \\ \end{align*}

Our one-line iterative procedure is:

\begin{align*} \mathbf{x}_{i + 1} = \mathbf{x}_i + \alpha_i \mathbf{p}_i \\ \end{align*}

Now we will slowly build up our procedure, step by step.

Recall that we want to make residual/gradient at the next iterate \(i + 1\) orthogonal to our direction at the current iterate \(i\).

\begin{align*} \mathbf{p}_i^T \mathbf{r}_{i + 1} = 0 \\ \end{align*}
\begin{align*} \implies \mathbf{p}_i^T H \mathbf{e}_{i + 1} = 0 \\ \end{align*}

Note that the error at the next iterate is H-conjugate to the previous direction. We will have more to say about this later.

Using our definitions, we have:

\begin{align*} \mathbf{x}_{i + 1} = \mathbf{x}_i + \alpha_i \mathbf{p}_i \\ \end{align*}
\begin{align*} \implies \mathbf{x}_{i + 1} - \mathbf{x}^* = \mathbf{x}_i - \mathbf{x}^* + \alpha_i \mathbf{p}_i \\ \end{align*}
\begin{align*} \implies \mathbf{e}_{i + 1} = \mathbf{e}_i + \alpha_i \mathbf{p}_i \\ \end{align*}

Substituting in the previous equation, we get:

\begin{align*} \implies \mathbf{p}_i^T H \mathbf{e}_{i + 1} = 0 \\ \end{align*}
\begin{align*} \implies \mathbf{p}_i^T H (\mathbf{e}_i + \alpha_i \mathbf{p}_i) = 0 \\ \end{align*}
\begin{align*} \implies \mathbf{p}_i^T H \mathbf{e}_i + \alpha_i \mathbf{p}_i^T H \mathbf{p}_i = 0 \\ \end{align*}
\begin{align*} \implies \mathbf{p}_i^T \mathbf{r}_i + \alpha_i \mathbf{p}_i^T H \mathbf{p}_i = 0 \\ \end{align*}
\begin{align*} \implies \alpha_i = - \frac{\mathbf{p}_i^T \mathbf{r}_i}{\mathbf{p}_i^T H \mathbf{p}_i}, \text{ where } \mathbf{r}_i = \nabla f(\mathbf{x}_i) = H \mathbf{x}_i + \mathbf{c} \\ \end{align*}

All of the terms in the right hand side of the above expression are known at the current iterate. This gives us our step length \(\alpha_i\), our goal #3!

Okay, we admit that it was arbitrary. Why should us wanting to orthogonalise the next residual to the current search direction give us a credible answer for the step length? So, let us try something else: given that we are standing at \(\mathbf{x}_i\), we want to minimise our objective \(f(\mathbf{x})\) at \(\mathbf{x}_{i + 1} = \mathbf{x}_i + \alpha_i \mathbf{p}_i\) over the direction \(\mathbf{p}_i\) (assume it is somehow already calculated). So, we can write:

\begin{align*} \underset{\mathbf{x}_{i + 1} \in \mathbb{R}^n}{\mathrm{min}} \: f(\mathbf{x}_{i + 1}) = \underset{\alpha_i \in \mathbb{R}}{\mathrm{min}} \: \phi(\alpha_i) = \frac{1}{2} (\mathbf{x}_i + \alpha_i \mathbf{p}_i)^T H (\mathbf{x}_i + \alpha_i \mathbf{p}_i) + \mathbf{c}^T (\mathbf{x}_i + \alpha_i \mathbf{p}_i) \\ \end{align*}
\begin{align*} \implies \underset{\alpha_i \in \mathbb{R}}{\mathrm{min}} \: \phi(\alpha_i) = \frac{1}{2} \mathbf{x}_i^T H \mathbf{x}_i + \frac{1}{2} \alpha_i^2 \mathbf{p}_i^T H \mathbf{p}_i + \alpha_i \mathbf{p}_i^T H \mathbf{x}_i + \mathbf{c}^T \mathbf{x}_i + \alpha_i \mathbf{c}^T \mathbf{p}_i \\ \end{align*}

Now comes our calculus attack: we will differentiate \(\phi(\alpha_i)\) with respect to \(\alpha_i\) and set the derivative to zero to find the directional minimum (Why? Because we know \(\mathbf{x}_i\), we assumed that we already know \(\mathbf{p}_i\), and we just need to find the minimum along this direction)

\begin{align*} \nabla_{\alpha_i} \phi(\alpha_i) = \frac{\partial \phi}{\partial \alpha_i} = \alpha_i \mathbf{p}_i^T H \mathbf{p}_i + \mathbf{p}_i^T H \mathbf{x}_i + \mathbf{c}^T \mathbf{p}_i \rightarrow 0 \\ \end{align*}
\begin{align*} \implies \alpha_i \mathbf{p}_i^T H \mathbf{p}_i + \mathbf{p}_i^T H \mathbf{x}_i + \mathbf{p}_i^T \mathbf{c} = 0 \\ \end{align*}
\begin{align*} \implies \alpha_i \mathbf{p}_i^T H \mathbf{p}_i + \mathbf{p}_i^T (H \mathbf{x}_i + \mathbf{c}) = 0 \\ \end{align*}
\begin{align*} \implies \alpha_i = - \frac{\mathbf{p}_i^T (H \mathbf{x}_i + \mathbf{c})}{\mathbf{p}_i^T H \mathbf{p}_i} \\ \end{align*}

which is nothing but (see definitions in Section II):

\begin{align*} \implies \alpha_i = - \frac{\mathbf{p}_i^T \mathbf{r}_i}{\mathbf{p}_i^T H \mathbf{p}_i} \text{ !} \\ \end{align*}

We have now arrived at the same step length as when we arbitrarily forced the next residual to be perpendicular to the current direction. Now, we know better: this step length actually minimises our objective along the current direction, and the former fact just comes about as a corollary! Further, we note that \(\mathbf{r}_{i + 1} \perp \mathbf{p}_i \implies \mathbf{p}_i^T \mathbf{r}_{i + 1} = \mathbf{p}_i^T H \mathbf{e}_{i + 1} = 0\), that is, direction - residual orthogonality implies direction - error H-conjugacy. In fact, we will make the following hand-waving argument: since the error at the next iterate is always H-conjugate to the current search direction, the current search direction is H-conjugate to all the previous search directions, and we don't ever step into a previously encountered direction again in the future, the next error is evermore H-conjugate to all the previous search directions by recursion. Which implies the residual is orthogonal to all the previous search directions! The rigorous proof of this statement is contained in the Expanding Subspace Minimization Theorem, which is presented in the next section.

Before we march ahead, let us establish a relationship between the residuals. We shall use this later in proving some theoretical guarantees. Using our error relationship, premultiply both sides of the equation by \(H\):

\begin{align*} \mathbf{e}_{i + 1} = \mathbf{e}_i + \alpha_i \mathbf{p}_i \\ \end{align*}
\begin{align*} \implies H \mathbf{e}_{i + 1} = H \mathbf{e}_i + \alpha_i H \mathbf{p}_i \\ \end{align*}
\begin{align*} \implies \mathbf{r}_{i + 1} = \mathbf{r}_i + \alpha_i H \mathbf{p}_i \\ \end{align*}

Next, we will try to find the directions \(\mathbf{p}_i\). Let us pick, from our available knowledge, n linearly independent vectors \(\{ \mathbf{u}_0, \mathbf{u}_1, \dots, \mathbf{u}_{n - 1} \}\), which we will call as the generator vectors. The easiest choice we can make is the coordinate axes, \(\{ \mathbf{e}_1, \mathbf{e}_2, \dots, \mathbf{e}_n \}\). To construct \(\mathbf{p}_i\), we will take each generator \(\mathbf{u}_i\) and peel off any components that are not H-conjugate to the entire set of previous \(i - 1\) \(\mathbf{p}\) vectors. To kickstart the process, set \(\mathbf{p}_0 = \mathbf{u}_0 = \mathbf{e}_1\) and for all \(i > 1\), iterate as:

\begin{align*} \mathbf{p}_i = \mathbf{u}_i + \sum_{k = 0}^{i - 1} \beta_{ik} \mathbf{p}_k, \: \beta_{ik} \text{ defined for } i > k \\ \end{align*}

To calculate the \(\beta_{ik}\) factors, we exploit the H-conjugacy property of \(\mathbf{p}_i\) directions. Transpose above equation and post-multiply with \(H \mathbf{p}_j, j \in [0, \dots, i - 1]\):

\begin{align*} \mathbf{p}_i^T H \mathbf{p}_j = \mathbf{u}_i^T H \mathbf{p}_j + \sum_{k = 0}^{i - 1} \beta_{ik} \mathbf{p}_k^T H \mathbf{p}_j \\ \end{align*}
\begin{align*} \implies 0 = \mathbf{u}_i^T H \mathbf{p}_j + \beta_{ij} \mathbf{p}_j^T H \mathbf{p}_j \\ \end{align*}
\begin{align*} \implies \beta_{ij} = - \frac{\mathbf{u}_i^T H \mathbf{p}_j}{\mathbf{p}_j^T H \mathbf{p}_j} \\ \end{align*}

Unfortunately, this means that to calculate each \(\mathbf{p}_i\) upto the n-th direction, all the previous \(H \mathbf{p}_j, j \in [0, \dots, i - 1]\) directions must be kept in storage. This is not very efficient. In fact, the total cost of generating all the search directions is \(\mathcal{O}(n^3)\)), making direct factorisation look appealing! This brings us to the topic of our next section: Conjugate Gradients (or Conjugate(d) Gradients / Conjugate(d) Residuals), a very efficient method of constructing Conjugate Directions.

But wait! Did we not mention before, this conjugate directions process will guarantee convergence in n-steps? Or will it, really? It sounds quite arbitrary. Are we full of ourselves? Why should it happen at all? Let's attack this notion with a mathematical analysis!

We will try to do this: first we will assume that the process indeed converges in n-steps, which implies, the initial error term \(\mathbf{e}_0\) is made up of n-vectors oriented along these directions \(\mathbf{p}_i\). Then, we shall use the H-conjugacy properties of the directions. If we find that, \(\mathbf{e}_n\), that is error after n-steps, is not zero, our big claim is grossly incorrect, and we will terribly fall from grace. However, if \(\mathbf{e}_0\) is indeed zero, we would have proven our n-step convergence assumption on stone and are right on target!

Without much ado, let's formulate the math:

\begin{align*} \mathbf{e}_0 = \sum_{j = 0}^{n - 1} \: \delta_j \mathbf{p}_j \\ \end{align*}

Premultiplying \(\mathbf{p}_k^T H\) and exploiting H-conjugacy, \(\mathbf{p}_k^T H \mathbf{p}_j = 0 \: \forall \: k \neq j\) :

\begin{align*} \mathbf{p}_k^T H \mathbf{e}_0 = \sum_{j = 0}^{n - 1} \: \delta_j \mathbf{p}_k^T H \mathbf{p}_j \\ \end{align*}
\begin{align*} \implies \mathbf{p}_k^T H \mathbf{e}_0 = \delta_k \mathbf{p}_k^T H \mathbf{p}_k \\ \end{align*}

Using the fact that \(\sum_{i = 0}^{k - 1} \: \alpha_i \mathbf{p}_k^T H \mathbf{p}_i = 0\),

\begin{align*} \implies \mathbf{p}_k^T H \mathbf{e}_0 + \sum_{i = 0}^{k - 1} \: \alpha_i \mathbf{p}_k^T H \mathbf{p}_i = \delta_k \mathbf{p}_k^T H \mathbf{p}_k \\ \end{align*}
\begin{align*} \implies \mathbf{p}_k^T H (\mathbf{e}_0 + \sum_{i = 0}^{k - 1} \: \alpha_i \mathbf{p}_i) = \delta_k \mathbf{p}_k^T H \mathbf{p}_k \\ \end{align*}

We also know that, \(\mathbf{e}_{i + 1} = \mathbf{e}_i + \alpha_i \mathbf{p}_i\) or \(\mathbf{e}_k = \mathbf{e}_{k - 1} + \alpha_{k - 1} \mathbf{p}_{k - 1}\). Recursively expanding this formula into a summation, we get, \(\mathbf{e}_k = \mathbf{e}_{k - 1} + \alpha_{k - 1} \mathbf{p}_{k - 1} = \mathbf{e}_0 + \sum_{i = 0}^{k - 1} \: \alpha_i \mathbf{p}_i\). Substituting this into the above expression:

\begin{align*} \implies \mathbf{p}_k^T H \mathbf{e}_k = \delta_k \mathbf{p}_k^T H \mathbf{p}_k \\ \end{align*}
\begin{align*} \implies \delta_k = \frac{\mathbf{p}_k^T \mathbf{r}_k}{\mathbf{p}_k^T H \mathbf{p}_k} \\ \end{align*}

This means:

\begin{align*} \delta_k = - \alpha_k \text{ ! } \\ \end{align*}

Moving ahead,

\begin{align*} \mathbf{e}_k = \mathbf{e}_0 + \sum_{i = 0}^{k - 1} \: \alpha_i \mathbf{p}_i \\ \end{align*}
\begin{align*} \mathbf{e}_k = \sum_{j = 0}^{n - 1} \: \delta_j \mathbf{p}_j - \sum_{i = 0}^{k - 1} \: \delta_i \mathbf{p}_i \\ \end{align*}
\begin{align*} \mathbf{e}_k = \sum_{j = 0}^{k - 1} \: \delta_j \mathbf{p}_j + \sum_{l = k}^{n - 1} \: \delta_l \mathbf{p}_l - \sum_{i = 0}^{k - 1} \: \delta_i \mathbf{p}_i \\ \end{align*}
\begin{align*} \mathbf{e}_k = \sum_{l = k}^{n - 1} \: \delta_l \mathbf{p}_l \\ \end{align*}

Substituting \(k = n\),

\begin{align*} \mathbf{e}_n = \sum_{l = n}^{n - 1} \: \delta_l \mathbf{p}_l = 0 \\ \end{align*}

because we do not have any directions beyond the (n - 1)-th iterate.

Therefore, the error at the n-th step is really zero, and we have arrived at this conclusion by just using the H-conjugacy property! Thus, we can safely state that this Conjugate Directions method indeed converges in n-steps, as we have been advertising. (Disclaimer: The problem has to be convex quadratic positive-definite!)

Okay, now we can move forward to the next sections in confidence and peace of mind.

Some Proofs on our Road to Enlightenment

Expanding Subspace Minimization (ESM) Theorem

Let \(\mathbf{x}_0 \in \mathbb{R}^n\). Given a set of H-conjugate \(\{ \mathbf{p}_k \}_0^{i - 1}\) directions, and \(\mathbf{x}_k = \mathbf{x}_{k} = \mathbf{x}_{k-1} + \alpha_{k - 1} \mathbf{p}_{k-1}\), the Expanding Subspace Minimization theorem states that,

\begin{align*} \mathbf{r}_k^T \mathbf{p}_i = 0 \: \forall \: i \in [\![ 0, k - 1 ]\!] \\ \end{align*}

and \(\mathbf{x}_k\) is the minimizer of:

\begin{align*} f(\mathbf{x}) = \frac{1}{2}\mathbf{x}^T H \mathbf{x} + c^T \mathbf{x} \\ \end{align*}
\begin{align*} \text{over the set } \{\mathbf{x} \mid \mathbf{x} = \mathbf{x}_0 + \mathrm{span}\{\mathbf{p}_0, \mathbf{p}_1,...\mathbf{p}_{k-1}\}\} \\ \end{align*}

The new residual \(\mathbf{r}_k\) is orthogonal to all the search directions used in the previous iterations (provided that the search directions are H-conjugate}.

Proof:

We will show that \(\tilde{\mathbf{x}} = \mathbf{x}_0 + \sigma_0^*\mathbf{p}_0 + \dots + \sigma_{k-1}^*\mathbf{p}_{k-1}\) minimizes \(f(\mathbf{x})\) over \(\{\mathbf{x} \mid \mathbf{x} = \mathbf{x}_0 + \mathrm{span}\{\mathbf{p}_0, \mathbf{p}_1,...\mathbf{p}_{k-1}\}\}\), if and only if \(\mathbf{r}(\tilde{\mathbf{x}})^T \mathbf{p}_i = 0, \: i = 0,...., k - 1\).

Let,

\begin{align*} h(\sigma) = f(\mathbf{x}_0 + \sigma_0^*\mathbf{p}_0 + \dots + \sigma_{k-1}^*\mathbf{p}_{k-1}) \\ \end{align*}
\begin{align*} \sigma = (\sigma_0, \sigma_1, ..., \sigma_{k-1})^T \\ \end{align*}

\(h(\sigma)\) is a strict convex quadratic. Therefore, at the unique minimiser \(\sigma^*\),

\begin{align*} \frac{\partial h (\sigma^*)}{\partial \sigma_i} = 0 \\ \end{align*}

By Chain Rule,

\begin{align*} \frac{\partial h (\sigma^*)}{\partial \tilde{\mathbf{x}}} \frac{\partial \tilde{\mathbf{x}}}{\partial \sigma_i} = 0 \\ \end{align*}
\begin{align*} \nabla f(\mathbf{x}_0 + \sigma_0^*\mathbf{p}_0 + \dots + \sigma_{k-1}^*\mathbf{p}_{k-1})^T \mathbf{p}_i = 0, \: i = 0,...,k-1 \\ \end{align*}
\begin{align*} \nabla f(\tilde{\mathbf{x}})^T \mathbf{p}_i = 0 \\ \end{align*}
\begin{align*} \mathbf{r}(\tilde{\mathbf{x}})^T \mathbf{p}_i = 0, \: i = 0,..., k-1 \\ \end{align*}

Hence, from the above, we can conclude that \(\tilde{\mathbf{x}}\) minimizes \(f(\mathbf{x})\) over \(\{\mathbf{x}, \mathbf{x} = \mathbf{x}_0 + \mathrm{span}\{\mathbf{p}_0, \mathbf{p}_1,...\mathbf{p}_{k-1}\}\}\), by the result that \(\mathbf{r}(\tilde{\mathbf{x}})^T \mathbf{p}_i = 0, \: i = 0,...., k - 1\).

Now, we will use induction to show that \(\mathbf{x}_k\) satisfies \(\mathbf{r}(\tilde{\mathbf{x}})^T \mathbf{p}_i = 0\), and therefore, \(\mathbf{x}_k = \tilde{\mathbf{x}}\).

For \(k = 1\), \(\mathbf{x}_1 = \mathbf{x}_0 + \alpha_0 \mathbf{p}_0\) minimizes \(f(\mathbf{x})\) along \(\mathbf{p}_0\) which implies \(\mathbf{r}_1^T \mathbf{p}_0 = 0\).

Let our induction hypothesis hold true for \(\mathbf{r}_{k-1}\), that is, \(\mathbf{r}_{k-1}^T \mathbf{p}_i = 0, \: \forall \: i = 0, \dots , k-2\). We will prove the orthogonality relationships for \(\mathbf{r}_{k}\). We know that,

\begin{align*} \mathbf{r}_{k} = \mathbf{r}_{k - 1} + \alpha_{k - 1} H \mathbf{p}_{k - 1} \\ \end{align*}
\begin{align*} \implies \mathbf{p}_{k-1}^T \mathbf{r}_{k} = \mathbf{p}_{k-1}^T \mathbf{r}_{k - 1} + \alpha_{k - 1} \mathbf{p}_{k - 1}^T H \mathbf{p}_{k - 1} \\ \end{align*}
\begin{align*} \implies \mathbf{r}_{k}^T \mathbf{p}_{k-1} = 0 \text{ after substituting the value of } \alpha_{k - 1} = - \frac{\mathbf{p}_{k - 1}^T \mathbf{r}_{k - 1}}{\mathbf{p}_{k - 1}^T H \mathbf{p}_{k - 1}} \\ \end{align*}

For the rest of the vectors \(\mathbf{p}_i, \: i = 0, \dots, k - 2\),

\begin{align*} \mathbf{r}_{k} = \mathbf{r}_{k - 1} + \alpha_{k - 1} H \mathbf{p}_{k - 1} \\ \end{align*}
\begin{align*} \implies \mathbf{p}_{i}^T \mathbf{r}_{k} = \mathbf{p}_{i}^T \mathbf{r}_{k - 1} + \alpha_{k - 1} \mathbf{p}_{i}^T H \mathbf{p}_{k - 1} \\ \end{align*}
\begin{align*} \implies \mathbf{r}_{k}^T \mathbf{p}_{i} = \mathbf{r}_{k - 1}^T \mathbf{p}_{i} + \alpha_{k - 1} \mathbf{p}_{k - 1} H \mathbf{p}_{i}^T \\ \end{align*}

The first term on the Right Hand Side vanishes due to our induction hypothesis. The second term vanishes due to the H-conjugacy of the \(\mathbf{p}_{i}\) vectors. Therefore,

\begin{align*} \mathbf{r}_{k}^T \mathbf{p}_{i} = 0 \: \forall \: i \in [\![ 0, k - 2 ]\!] \\ \end{align*}

Combining both of the above results, we get,

\begin{align*} \mathbf{r}_{k}^T \mathbf{p}_{i} = 0 \: \forall \: i \in [\![ 0, k - 1 ]\!] \\ \end{align*}

and therefore, \(\mathbf{x}_k\) is the minimizer of:

\begin{align*} f(\mathbf{x}) = \frac{1}{2}\mathbf{x}^T H \mathbf{x} + c^T \mathbf{x} \\ \end{align*}

over \(\{\mathbf{x} \mid \mathbf{x} = \mathbf{x}_0 + \mathrm{span}\{\mathbf{p}_0, \mathbf{p}_1,...\mathbf{p}_{k-1}\}\}\). Hence, proved.

Connection to Krylov Subspaces

The Expanding Subspace Minimisation (ESM) Theorem proves two things: the next iterate \(\mathbf{x}_i\) minimizes our objective \(f(\mathbf{x})\) over the entire subspace \(\mathbf{x}_0 + \underset{k}{\mathrm{span}}\{ \mathbf{p}_k \}_0^{i - 1}\); and the next residual \(\mathbf{r}_i\) is orthogonal to all the previous search directions \(\{ \mathbf{p}_k \}_0^{i - 1}\), that is,

\begin{align*} \mathbf{r}_i^T \mathbf{p}_k = 0 \: \forall \: k \in [\![ 0, i - 1 ]\!] \\ \end{align*}

Let us denote this subspace as \(\mathcal{D}_i = \{ \mathbf{p}_k \}_0^{i - 1}\).

In other words, the residual \(\mathbf{r}_i\) is orthogonal to the entire subspace \(\mathcal{D}_i\), and therefore, any vector contained in \(\mathcal{D}_i\) will be orthogonal to \(\mathbf{r}_i\)!

\begin{align*} \mathbf{r}_i \perp \mathcal{D}_i \\ \end{align*}

The image below demonstrates this fact for \(\mathcal{D}_2\).

Here we attempt to analyse a few more fundamental issues.

In plain English, what the first part of the ESM theorem is saying is that given a starting point \(\mathbf{x}_0\), we keep on enlarging the expanding-subspace \(\mathcal{D}_i\) by one dimension in each step, and we evermore keep getting closer to the solution after gaining that extra degree of freedom of movement in that step. As soon as \(\mathcal{D}_i\) becomes \(\mathbb{R}^n\), the error gets driven to zero and we have reached our destination!

Since we have mentioned in the previous section that the directions \(\mathbf{p}\) are built from the linearly independent set of vectors \(\mathbf{u}\), that is,

\begin{align*} \mathbf{p}_i = \mathbf{u}_i + \sum_{k = 0}^{i - 1} \beta_{ik} \mathbf{p}_k \\ \end{align*}

..it implies that the space spanned by \(\{ \mathbf{p}_k \}_0^{i - 1}\) is the same as the space spanned by \(\{ \mathbf{u}_k \}_0^{i - 1}\) by linearity. That is, \(\mathcal{D}_i = \{ \mathbf{u}_k \}_0^{i - 1}\). Since, \(\mathbf{r}_i \perp \mathcal{D}_i\), this means, the next residual \(\mathbf{r}_i\) is also orthogonal to all the previous and current generator \(\{ \mathbf{u}_k \}_0^{i - 1}\) vectors lying in the space \(\mathcal{D}_i\)! Mathematically, by first transposing the above equation,

\begin{align*} \mathbf{p}_i^T = \mathbf{u}_i^T + \sum_{k = 0}^{i - 1} \beta_{ik} \mathbf{p}_k^T \\ \end{align*}

Let \(i < j\) or \(i \in [\![ 0, j - 1 ]\!]\). Then,

\begin{align*} \mathbf{p}_i^T \mathbf{r}_j = \mathbf{u}_i^T \mathbf{r}_j + \sum_{k = 0}^{i - 1} \beta_{ik} \mathbf{p}_k^T \mathbf{r}_j \\ \end{align*}

Using subspace minimization theorem,

\begin{align*} 0 = \mathbf{u}_i^T \mathbf{r}_j + 0 \\ \end{align*}
\begin{align*} \implies \mathbf{r}_j^T \mathbf{u}_i = 0 \: \forall \: i \in [\![ 0, j - 1 ]\!] \\ \end{align*}

Hence, we see why it is true.

Now, comes the core idea of Conjugate Gradient: Instead of using the coordinate axes \(\{ \mathbf{e}_1, \mathbf{e}_2, \dots, \mathbf{e}_n \}\) as the generator vectors \(\{ \mathbf{u}_0, \mathbf{u}_1, \dots, \mathbf{u}_{n - 1} \}\) for the conjugated-construction of our search directions \(\mathbf{p}_i\), we will instead use the negative residuals/gradients \(\{ - \mathbf{r}_0, - \mathbf{r}_1, \dots, - \mathbf{r}_{n - 1} \}\) as the generator vectors! That is, \(\mathbf{u}_i = - \mathbf{r}_i\). This is precisely the reason why we have emphasized on the term, Conjugate(d) Gradients. Because the residuals/gradients are themselves not H-conjugate, but the n H-conjugate directions that we need for our algorithm are constructed out of them.

While this choice seems arbitrary, we will also see how this choice leads to the efficiency and some more desirable attributes of the CG algorithm. Since \(\mathbf{p}_0 = \mathbf{u}_0 = -\mathbf{r}_0 = -\nabla f(\mathbf{x}_0)\), the first direction of the Conjugate Gradient process is also the direction of Steepest Descent.

We stated before that,

\begin{align*} \implies \mathbf{r}_j^T \mathbf{u}_i = 0 \: \forall \: i \in [\![ 0, j - 1 ]\!] \\ \end{align*}

Replace \(\mathbf{u}_i = -\mathbf{r}_i\) in the above and we shall get,

\begin{align*} \implies - \mathbf{r}_j^T \mathbf{r}_i = 0 \: \forall \: i \in [\![ 0, j - 1 ]\!] \\ \end{align*}

(Dropping the negative sign, and replacing the indices \(i\) by \(k\) and \(j\) by \(i\) with no difference)

\begin{align*} \implies \mathbf{r}_i^T \mathbf{r}_k = 0 \: \forall \: k \in [\![ 0, i - 1 ]\!] \\ \end{align*}

Which means the next residual/gradient is orthogonal to all the previous residuals/gradients. Since \(\mathbf{u}_i\) (generator) vectors have to be linearly independent for generating our directions, by selecting \(\mathbf{u}_i = -\mathbf{r}_i\) we are not doing any mistake since the mutual orthogonality of \(\mathbf{r}_i \: \forall \: i \in [\![ 0, n - 1 ]\!]\) guarantees their linear independence. Contrast this with Steepest Descent, where a lot of residuals/gradients were repeated directions and hence not linearly independent.

Therefore, the space spanned by directions \(\{ \mathbf{p}_k \}_0^{i - 1}\) is equal to the space spanned by the residuals \(\{ \mathbf{r}_k \}_0^{i - 1}\) (since \(\{ \mathbf{p}_k \}_0^{i - 1} = \{ \mathbf{u}_k \}_0^{i - 1}\)).

\begin{align*} \mathcal{D}_i = \underset{k}{\mathrm{span}}\{ \mathbf{p}_k \}_0^{i - 1} = \underset{k}{\mathrm{span}}\{ \mathbf{u}_k \}_0^{i - 1} = \underset{k}{\mathrm{span}}\{ \mathbf{r}_k \}_0^{i - 1} \\ \end{align*}

Further, let us develop one more statement:

\begin{align*} \mathbf{p}_i^T \mathbf{r}_i = \mathbf{u}_i^T \mathbf{r}_i + \sum_{k = 0}^{i - 1} \beta_{ik} \mathbf{p}_k^T \mathbf{r}_i \\ \end{align*}
\begin{align*} \implies \mathbf{p}_i^T \mathbf{r}_i = \mathbf{u}_i^T \mathbf{r}_i = - \mathbf{r}_i^T \mathbf{r}_i \\ \end{align*}

We know that,

\begin{align*} \mathbf{r}_i = \mathbf{r}_{i - 1} + \alpha_{i - 1} H \mathbf{p}_{i - 1} \\ \end{align*}

Since, \(\mathbf{r}_i\) lies outside \(\mathcal{D}_i\) from the subspace identity, and \(\mathbf{r}_{i - 1} \in \mathcal{D}_i\), therefore, the \(H \mathbf{p}_{i - 1}\) vector has to lie outside the space spanned by \(\mathcal{D}_i\). From this, we can write: \(\mathbf{p}_{i - 1} \in \mathcal{D}_i \implies H \mathbf{p}_{i - 1} \in H \mathcal{D}_i\). Therefore, each subspace \(\mathcal{D}_{i + 1} = \underset{k}{\mathrm{span}}\{ \{ \mathbf{r}_k \}_0^{i - 1}, \mathbf{r}_i \}\) is constructed from the union of \(\mathcal{D}_i\) and \(H \mathcal{D}_i\).

Starting from the first subspace,

\begin{align*} \mathcal{D}_1 = \text{span}\{ \mathbf{p}_0 \} \\ \end{align*}

The second subspace is,

\begin{align*} \mathcal{D}_2 = \text{span}\{ \mathbf{p}_0 \} + H \text{span}\{ \mathbf{p}_0 \} \\ \end{align*}
\begin{align*} \mathcal{D}_2 = \text{span}\{ \mathbf{p}_0, H \mathbf{p}_0 \} \\ \end{align*}

The third subspace is,

\begin{align*} \mathcal{D}_3 = \text{span}\{ \mathbf{p}_0, H \mathbf{p}_0 \} + H \text{span}\{ \mathbf{p}_0, H \mathbf{p}_0 \} \\ \end{align*}
\begin{align*} \mathcal{D}_3 = \text{span}\{ \mathbf{p}_0, H \mathbf{p}_0, H (H \mathbf{p}_0) \} \\ \end{align*}

and therefore, by recursion, the i-th subspace is,

\begin{align*} \mathcal{D}_i = \text{span}\{ \mathbf{p}_0, H \mathbf{p}_0, H (H \mathbf{p}_0), H (H (H \mathbf{p}_0)), \dots, H^{i - 1} \mathbf{p}_0\} \\ \end{align*}
\begin{align*} \implies \mathcal{D}_i = \text{span}\{ \mathbf{p}_0, H \mathbf{p}_0, H^2 \mathbf{p}_0, H^3 \mathbf{p}_0, \dots, H^{i - 1} \mathbf{p}_0\} \\ \end{align*}

Since, \(\mathcal{D}_i = \underset{k}{\mathrm{span}}\{ \mathbf{p}_k \}_0^{i - 1} = \underset{k}{\mathrm{span}}\{ \mathbf{r}_k \}_0^{i - 1}\) by our generator choice, by the same arguments,

\begin{align*} \implies \mathcal{D}_i = \text{span}\{ \mathbf{r}_0, H \mathbf{r}_0, H^2 \mathbf{r}_0, H^3 \mathbf{r}_0, \dots, H^{i - 1} \mathbf{r}_0\} \\ \end{align*}

This subspace has a special name: Krylov Subspace, denoted by \(\mathcal{K}_i\). This particular subspace, which we constructed out of choosing the generator vectors \(\mathbf{u}_i\) as the residuals/gradients \(\mathbf{r}_i\), is in fact the core property that makes our Conjugate Gradient method a highly efficient one. So why is that? If we look carefully, we can observe that the Krylov Subspace \(\mathcal{D}_{i + 1}\) is the union of \(\mathcal{D}_i\) and \(H \mathcal{D}_i\) by construction. Since \(\mathbf{r}_{i + 1} \perp \mathcal{D}_{i + 1}\) as proven by the ESM Theorem, we can write,

\begin{align*} \mathbf{r}_{i + 1} \perp \mathcal{D}_{i + 1} \\ \end{align*}
\begin{align*} \mathbf{r}_{i + 1} \perp \left( \mathcal{D}_i \bigcup H \mathcal{D}_i \right) \\ \end{align*}

resulting in,

\begin{align*} \mathbf{r}_{i + 1} \perp \mathcal{D}_i \\ \end{align*}

and,

\begin{align*} \mathbf{r}_{i + 1} \perp H \mathcal{D}_i \\ \end{align*}

which is a funny result. Simplifying the above expression with \(\mathcal{D}_i = \underset{k}{\mathrm{span}}\{ \mathbf{p}_k \}_0^{i - 1} = \mathrm{span}\{ \mathbf{p}_0, \mathbf{p}_1, \mathbf{p}_2, \dots, \mathbf{p}_{i - 1} \}\),

\begin{align*} \mathbf{r}_{i + 1} \perp H \mathrm{span}\{ \mathbf{p}_0, \mathbf{p}_1, \mathbf{p}_2, \dots, \mathbf{p}_{i - 1} \} \\ \end{align*}
\begin{align*} \mathbf{r}_{i + 1} \perp \mathrm{span}\{ H \mathbf{p}_0, H \mathbf{p}_1, H \mathbf{p}_2, \dots, H \mathbf{p}_{i - 1} \} \\ \end{align*}
\begin{align*} \mathbf{r}_{i + 1}^T H \mathbf{p}_k = 0 \: \forall \: k \in [\![ 0, i - 1 ]\!] \\ \end{align*}

What this is saying is that \(\mathbf{r}_{i + 1}\) is H-conjugate to all the previous search directions except \(\mathbf{p}_i\) (\(\mathbf{r}_{i + 1}\) is also orthogonal to all the previous search directions as shown above, which is the reason this fact seems funny). Recall that we said in the previous section, to construct a new direction \(\mathbf{p}_{i + 1}\), we want to take a linearly independent generator \(\mathbf{u}_{i + 1}\) and peel off any of its components that are not H-conjugate to the set of directions \(\{ \mathbf{p}_k \}_0^i\).

\begin{align*} \mathbf{p}_{i + 1} = \mathbf{u}_{i + 1} + \sum_{k = 0}^{i} \beta_{(i + 1),k} \mathbf{p}_k \\ \end{align*}

Since our generator directions are the residuals/gradients, replace \(\mathbf{u}_{i + 1} = - \mathbf{r}_{i + 1}\),

\begin{align*} \mathbf{p}_{i + 1} = - \mathbf{r}_{i + 1} + \sum_{k = 0}^{i} \beta_{(i + 1),k} \mathbf{p}_k \\ \end{align*}

BUT! We just proved that \(\mathbf{r}_{i + 1}^T H \mathbf{p}_k = 0 \: \forall \: k \in [\![ 0, i - 1 ]\!]\)! It means there are no components in the vector \(\mathbf{r}_{i + 1}\) that are not H-conjugate to the set of directions \(\{ \mathbf{p}_k \}_0^{i - 1}\), and therefore we do not have to bother about removing such components from \(\mathbf{r}_{i + 1}\) as we have eliminated their presence! The only direction with which \(\mathbf{r}_{i + 1}\) is not H-conjugate is \(\mathbf{p}_i\). This greatly simplifies our job: we will simply retain the \(\mathbf{p}_i\) term and drop all other terms (as they are unnecessary):

\begin{align*} \mathbf{p}_{i + 1} = - \mathbf{r}_{i + 1} + \beta_{(i + 1),i} \mathbf{p}_i \\ \end{align*}

Now we simply replace \(\beta_{(i + 1),i} = \beta_{i + 1}\) and voila! Out comes our final form of a new H-conjugate direction:

\begin{align*} \mathbf{p}_{i + 1} = - \mathbf{r}_{i + 1} + \beta_{i + 1} \mathbf{p}_i \\ \end{align*}

To calculate \(\beta_{i + 1}\), transpose the above equation and post-multiply by \(H \mathbf{p}_i\),

\begin{align*} \mathbf{p}_{i + 1}^T H \mathbf{p}_i = - \mathbf{r}_{i + 1}^T H \mathbf{p}_i + \beta_{i + 1} \mathbf{p}_i^T H \mathbf{p}_i \\ \end{align*}
\begin{align*} \implies \beta_{i + 1} = \frac{\mathbf{r}_{i + 1}^T H \mathbf{p}_i}{\mathbf{p}_i^T H \mathbf{p}_i} \\ \end{align*}

Using \(\mathbf{r}_{i + 1} = \mathbf{r}_{i} + \alpha_{i} H \mathbf{p}_{i}\),

\begin{align*} \mathbf{r}_{i + 1}^T \mathbf{r}_{i + 1} = \mathbf{r}_{i + 1}^T \mathbf{r}_{i} + \alpha_{i} \mathbf{r}_{i + 1}^T H \mathbf{p}_{i} \\ \end{align*}
\begin{align*} \implies \mathbf{r}_{i + 1}^T \mathbf{r}_{i + 1} = \alpha_{i} \mathbf{r}_{i + 1}^T H \mathbf{p}_{i} \\ \end{align*}

And,

\begin{align*} \mathbf{p}_{i}^T \mathbf{r}_{i + 1} = \mathbf{p}_{i}^T \mathbf{r}_{i} + \alpha_{i} \mathbf{p}_{i}^T H \mathbf{p}_{i} \\ \end{align*}
\begin{align*} \implies 0 = - \mathbf{r}_{i}^T \mathbf{r}_{i} + \alpha_{i} \mathbf{p}_{i}^T H \mathbf{p}_{i} \\ \end{align*}
\begin{align*} \implies \mathbf{r}_{i}^T \mathbf{r}_{i} = \alpha_{i} \mathbf{p}_{i}^T H \mathbf{p}_{i} \\ \end{align*}

Substitute in the expression for \(\beta_{i + 1}\),

\begin{align*} \beta_{i + 1} = \frac{\alpha_{i} \mathbf{r}_{i + 1}^T H \mathbf{p}_i}{\alpha_{i} \mathbf{p}_i^T H \mathbf{p}_i} \\ \end{align*}
\begin{align*} \implies \beta_{i + 1} = \frac{\mathbf{r}_{i + 1}^T \mathbf{r}_{i + 1}}{\mathbf{r}_{i}^T \mathbf{r}_{i}} \\ \end{align*}

Now we are onto something! We neither need to store all directions in memory nor need them to calculate each new direction. This discovery firmly made the CG algorithm one of the mainstays of modern scientific computing.

As as final step, let us prove one more important thing: that the direction \(\mathbf{p}_{i + 1}\) generated by just \(\mathbf{r}_{i + 1}\) and \(\mathbf{p}_i\) at the i-th step is indeed H-conjugate to the entire set \(\{ \mathbf{p}_k \}_0^{i - 1}\) not considered in the formula. Assume that the set upto to the i-th iterate is already H-conjugate.

First, we transpose the following equation:

\begin{align*} \mathbf{p}_{i + 1} = - \mathbf{r}_{i + 1} + \beta_{i + 1} \mathbf{p}_i \\ \end{align*}
\begin{align*} \mathbf{p}_{i + 1}^T = - \mathbf{r}_{i + 1}^T + \beta_{i + 1} \mathbf{p}_i^T \\ \end{align*}

Post-multiply with \(H \mathbf{p}_{k}\), \(k \in [\![ 0, i - 1 ]\!]\),

\begin{align*} \mathbf{p}_{i + 1}^T H \mathbf{p}_{k} = - \mathbf{r}_{i + 1}^T H \mathbf{p}_{k} + \beta_{i + 1} \mathbf{p}_i^T H \mathbf{p}_{k} \\ \end{align*}

Using \(\mathbf{r}_{i + 1}^T H \mathbf{p}_k = 0 \: \forall \: k \in [\![ 0, i - 1 ]\!]\) and the fact that the set \(\{ \mathbf{p}_k \}_0^{i}\) is H-conjugate, both the right hand terms vanish. Therefore,

\begin{align*} \mathbf{p}_{i + 1}^T H \mathbf{p}_{k} = 0 \: \forall \: k \in [\![ 0, i - 1 ]\!] \\ \end{align*}

Hence, proved.

What we have done so far achieves both goal #2 and goal #1. Before presenting the Conjugate Gradient algorithm in the next section, let us summarise the most important facts from this section:

\begin{align*} \mathbf{r}_i^T \mathbf{p}_k = 0 \: \forall \: k \in [\![ 0, i - 1 ]\!] \\ \end{align*}
\begin{align*} \mathbf{r}_i^T \mathbf{r}_k = 0 \: \forall \: k \in [\![ 0, i - 1 ]\!] \\ \end{align*}
\begin{align*} \mathbf{p}_{i}^T H \mathbf{p}_{k} = 0 \: \forall \: k \in [\![ 0, i - 1 ]\!] \\ \end{align*}
\begin{align*} \mathbf{p}_i^T \mathbf{r}_i = \mathbf{u}_i^T \mathbf{r}_i = - \mathbf{r}_i^T \mathbf{r}_i \\ \end{align*}
\begin{align*} \mathcal{D}_i = \mathcal{K}_i = \mathrm{span}\{ \mathbf{r}_0, \mathbf{r}_1, \mathbf{r}_2, \dots, \mathbf{r}_{i - 1} \} = \mathrm{span}\{ \mathbf{p}_0, \mathbf{p}_1, \mathbf{p}_2, \dots, \mathbf{p}_{i - 1} \} \\ = \text{span}\{ \mathbf{r}_0, H \mathbf{r}_0, H^2 \mathbf{r}_0, H^3 \mathbf{r}_0, \dots, H^{i - 1} \mathbf{r}_0\} \\ \end{align*}

Conjugate Gradients: A Method of Conjugate Directions

All of our machinery is in place now. We collect the necessary facts in an iterative loop; the simplicity and terseness of the algorithm completely belies the conceptual buildup we went through in the previous sections. So, without any further ado, let's present the Conjugate Gradient algorithm in one piece now.

  1. Given, \(\mathbf{x}_0\).
  2. Set \(\mathbf{r}_0 \leftarrow H \mathbf{x}_0 + \mathbf{c}\), \(\mathbf{p}_0 \leftarrow - \mathbf{r}_0\) and \(i \leftarrow 0\).
  3. while \(\lVert \mathbf{r}_i \rVert_2 \neq 0\), do:
  4. Step Length:
\begin{align*} \alpha_i = \frac{\mathbf{r}_i^T \mathbf{r}_i}{\mathbf{p}_i^T H \mathbf{p}_i} \\ \end{align*}
  1. Next Iterate:
\begin{align*} \mathbf{x}_{i + 1} = \mathbf{x}_i + \alpha_i \mathbf{p}_i \\ \end{align*}
  1. Next Residual:
\begin{align*} \mathbf{r}_{i + 1} = \mathbf{r}_i + \alpha_i H \mathbf{p}_i \\ \end{align*}
  1. Direction Update Factor:
\begin{align*} \beta_{i + 1} = \frac{\mathbf{r}_{i + 1}^T \mathbf{r}_{i + 1}}{\mathbf{r}_{i}^T \mathbf{r}_{i}} \\ \end{align*}
  1. Next Direction:
\begin{align*} \mathbf{p}_{i + 1} = - \mathbf{r}_{i + 1} + \beta_{i + 1} \mathbf{p}_i \\ \end{align*}
  1. Loop Counter Update:
\begin{align*} i \leftarrow i + 1 \\ \end{align*}
  1. end while

Computational Complexity:

The dominant costs of the above algorithm is in the Matrix-Vector multiplication step: \(H \mathbf{p}_i\).

For a dense matrix \(H\), the Matrix-Vector multiplication costs, with \(H \rightarrow n \times n\) and \(\mathbf{p}_i \rightarrow n \times 1\), \(\mathcal{O}(n^2)\).

For a sparse matrix \(H\), Matrix-Vector multiplication costs, \(\mathcal{O}(m)\), where \(m\) is the non-zero elements in \(H\).

If we want the initial residual \(H \mathbf{r}_0\) to decrease by a factor \(\epsilon\), say:

\(\| \mathbf{r}_k\| \leq \epsilon \| \mathbf{r}_0 \|\), where \(\mathbf{r}_k\) and \(\mathbf{r}_0\) are the residuals in the \(k^{th}\) and the \(0^{th}\) iterations, then the Number of Iterations, \(n_{iter}\) required for the following algorithms are:

Steepest Descent: \(n_{iter} \leq \frac{1}{2}\mathcal{K}\ln{\frac{1}{\epsilon}}\).

Conjugate Gradient \(n_{iter} \leq \frac{1}{2}\sqrt{\mathcal{K}}\ln{\frac{2}{\epsilon}}\).

where,

\(\mathcal{K}\) is the condition number of \(H\).

If we have a \(d\)-dimensional domain, to solve the boundary value problem, the condition number \(\mathcal{K}\) of the matrix, obtained by discretizing the second order elliptic boundary value problem, is of the order \(\mathcal{O}(n^\frac{2}{d})\), and the number of non-zero entries is of \(\mathcal{O}(n)\).

The Time Complexities for the two methods in this problem are as:

Steepest Descent: \(\mathcal{O}(m\mathcal{K})\)

Conjugate Gradient: \(\mathcal{O}(m\sqrt{\mathcal{K}})\)

For a 2D problem and \(m\) \(\epsilon\) \(\mathcal{O}(n)\), we have the following:

S.D: Time complexity: \(\mathcal{O}(n^2)\)

C.G: Time complexity: \(\mathcal{O}(n^\frac{3}{2})\)

Both methods have a space complexity of \(\mathcal{O}(m)\).

Convergence of CG:

In exact arithmetic, the conjugate gradient method will terminate at the solution in at most \(n\) iterations. When the distribution of the eigenvalues of \(H\) has certain favourable features, the algorithm will identify the solution in many fewer than \(n\) iterations.

If \(H\) has only \(r\) distinct eigenvalues, then the CG iteration will terminate at the solution in at most \(r\) iterations.

It is generally true that if the eigenvalues occur in \(r\) distinct clusters, the CG iterates will approximately solve the problem in about \(r\) steps.

Pre conditioned CG:

When the condition number \(\mathcal{K}\) of \(H\) is really high, then we run into issues. We introduce a technique called as pre-conditioning.

Here we transform the problem \(Hx = -c\) with the inverse of \(M\) where \(M\) is an SPD and invertible matrix, chosen by the user depending on the problem at hand:

\begin{align*} M^{-1} H \mathbf{x} = -M^{-1} \mathbf{c} \\ \end{align*}

We need: \(\mathcal{K}(M^{-1}H) \ll \mathcal{K}(H)\) or the eigenvalues of \(M^{-1}H\) are clustered together, thereby ensuring faster convergence to the actual solution in \(r\) steps, if there are \(r\) clusters.

Most of the standard solver packages use pre-conditioning for faster convergence.

Naive Code Implementation

The following class implements the reference Conjugate Gradient algorithm without preconditioning, for educational purposes only. Do not use it on your actual code!

import numpy as np

class quadratic_form:
    """
    Solve a linear system Hx = -c using a quadratic formulation.
    Uses the Conjugate Gradient method.
    """

    def __init__(self, H: np.ndarray, c: np.ndarray, rhs_sign: np.int64) -> None:
        """
        __init__: initialise the CG class.

        Input data = H, x, c.

        Parameters
        ----------
        H : np.ndarray
            System Matrix.
        c : np.ndarray
            rhs
        rhs_sign : np.int64
            Sign of rhs. If system is Hx = c, set rhs_sign = 1, else if Hx = - c, set rhs_sign = -1.
        """
        self.H = H
        self.c = c
        self.res_sgn = -rhs_sign
        self.rows, self.cols = H.shape

    def cgmin(
        self, x0: np.ndarray, maxiter: np.int64 = 2000, tol: np.float64 = 1.0e-8, debug: bool = False
    ) -> np.ndarray:
        """
        cgmin: Minimize positive definite quadratic form with conjugate gradient method.

        This routine uses the un-preconditioned efficient version of CG.

        Parameters
        ----------
        x0 : np.ndarray
            Initial Guess Vector

        maxiter : np.int64
            Maximum number of allowed iterations

        tol: np.float64
            Tolerance

        debug : bool
            Print iteration steps
        """
        x = x0  # initial guess
        r: np.ndarray = self.H @ x0 + self.res_sgn * self.c  # initial residual
        p: np.ndarray = -r  # initial direction
        i: np.int64 = 0

        rtr = r.T @ r
        while np.linalg.norm(r) > tol and i < maxiter:
            Hpi = self.H @ p
            alpha = rtr / (p.T @ Hpi)
            x = x + alpha * p
            r = r + alpha * Hpi
            rnr = r.T @ r
            if np.linalg.norm(r) < tol:
                break
            beta = rnr / rtr
            rtr = rnr
            p = -r + beta * p
            i += 1
            if debug:
                print(f"x is {x} at iteration #{i}")

        if i == maxiter:
            print("Max Iterations reached. Terminating procedure...")
            print(f"Error norm attained: {np.linalg.norm(r)}")
        else:
            print(f"Solution reached in {i} iterations.")
            print(f"Final Error Norm at solution: {np.linalg.norm(r)}")

        return x

The following sample code uses the above class to solve a symmetric positive-definite linear system:

A = 500 * np.random.randn(500, 500)
H = A.T @ A
c = 500 * np.random.randn(500)
x0 = np.ones(c.shape)

qform = quadratic_form(H, c, 1)
x = qform.cgmin(x0, maxiter=5000, tol=1e-8, debug=False)

print("Solution is :")
print(x)

Available Software

A high-quality, expert-developed Python implementation of the Conjugate Gradient method is available in the Scipy library as the scipy.sparse.linalg.cg function. See the documentation for more details.

Well engineered, high performance C language implementations of Conjugate Gradient are available in the Intel Math Kernel Library. The following links contain more information:

  1. Iterative Sparse Solvers based on Reverse Communication Interface
  2. CG Interface Description
  3. dcg_init
  4. dcg_check
  5. dcg
  6. dcg_get
  7. Preconditioners based on Incomplete LU Factorization Technique

Non-Linear CG

If our objective function is not convex quadratic as we were dealing with till now, we will have to adapt our current way to fit it into a non-linear functions.

We were solving exactly for \(\alpha_k\) till now, but in the nonlinear case, we have to use the Backtracking line-search scheme to calculate \(\alpha\).

Instead of the residual \(r_k\), we will use \(\nabla f_k\), the gradient of the objective function at the current iterate.

\(\beta_{k+1}\) has to be recalculated in terms of \(\nabla f_k\) and the new search direction \(p_k\).

\(\beta_{k}\) can be calculated by the following update equations:

Fletcher - Reeves Method

\begin{align*} \beta_{FR}^k = \frac{\nabla f_k^T \nabla f_k}{\nabla f_{k-1}^T \nabla f_{k-1}} \\ \end{align*}

Polak Ribere Method:

\begin{align*} \beta_{PR}^k = \frac{\nabla f_k^T (\nabla f_k - \nabla f_{k-1})}{\nabla f_{k-1}^T \nabla f_{k-1}} \\ \end{align*}

Hestenes-Steifel Method:

\begin{align*} \beta_{HS}^k = \frac{\nabla f_k^T (\nabla f_k - \nabla f_{k-1})}{(\nabla f_k - \nabla f_{k-1})^T p_{k-1}}\\ \end{align*}