A major tool in numerical linear algebra is to factor a given matrix into terms that are individually easier to deal with than the original. In this section we derive a means to express a square matrix using triangular factors, which will allow us to solve a linear system using forward and backward substitution.
2.4.1 Outer products ¶ Our derivation of the factorization hinges on an expression of matrix products in terms of vector outer products. If u ∈ R m \mathbf{u}\in\real^m u ∈ R m and v ∈ R n \mathbf{v}\in\real^n v ∈ R n , then the outer product of these vectors is the m × n m\times n m × n matrix
u v T = [ u 1 v 1 u 1 v 2 ⋯ u 1 v n u 2 v 1 u 2 v 2 ⋯ u 2 v n ⋮ ⋮ ⋮ u m v 1 u m v 2 ⋯ u m v n ] . \mathbf{u} \mathbf{v}^T =
\begin{bmatrix}
u_1 v_1 & u_1 v_2 & \cdots & u_1 v_n \\u_2 v_1 & u_2 v_2 & \cdots & u_2 v_n \\ \vdots & \vdots & & \vdots \\ u_m v_1 & u_m v_2 & \cdots & u_m v_n
\end{bmatrix}. u v T = ⎣ ⎡ u 1 v 1 u 2 v 1 ⋮ u m v 1 u 1 v 2 u 2 v 2 ⋮ u m v 2 ⋯ ⋯ ⋯ u 1 v n u 2 v n ⋮ u m v n ⎦ ⎤ . We illustrate the connection of outer products to matrix multiplication by a small example.
According to the usual definition of matrix multiplication,
[ 4 − 1 − 3 5 − 2 6 ] [ 2 − 7 − 3 5 ] = [ ( 4 ) ( 2 ) + ( − 1 ) ( − 3 ) ( 4 ) ( − 7 ) + ( − 1 ) ( 5 ) ( − 3 ) ( 2 ) + ( 5 ) ( − 3 ) ( − 3 ) ( − 7 ) + ( 5 ) ( 5 ) ( − 2 ) ( 2 ) + ( 6 ) ( − 3 ) ( − 2 ) ( − 7 ) + ( 6 ) ( 5 ) ] . \begin{align*}
\small
\begin{bmatrix}
4 & -1 \\ -3 & 5 \\ -2 & 6
\end{bmatrix}
\begin{bmatrix}
2 & -7 \\ -3 & 5
\end{bmatrix}
& =
\small
\begin{bmatrix}
(4)(2) + (-1)(-3) & (4)(-7) + (-1)(5) \\
(-3)(2) + (5)(-3) & (-3)(-7) + (5)(5) \\
(-2)(2) + (6)(-3) & (-2)(-7) + (6)(5)
\end{bmatrix}.
\end{align*} [ 4 − 3 − 2 − 1 5 6 ] [ 2 − 3 − 7 5 ] = [ ( 4 ) ( 2 ) + ( − 1 ) ( − 3 ) ( − 3 ) ( 2 ) + ( 5 ) ( − 3 ) ( − 2 ) ( 2 ) + ( 6 ) ( − 3 ) ( 4 ) ( − 7 ) + ( − 1 ) ( 5 ) ( − 3 ) ( − 7 ) + ( 5 ) ( 5 ) ( − 2 ) ( − 7 ) + ( 6 ) ( 5 ) ] . If we break this up into the sum of two matrices, however, each is an outer product.
= [ ( 4 ) ( 2 ) ( 4 ) ( − 7 ) ( − 3 ) ( 2 ) ( − 3 ) ( − 7 ) ( − 2 ) ( 2 ) ( − 2 ) ( − 7 ) ] + [ ( − 1 ) ( − 3 ) ( − 1 ) ( 5 ) ( 5 ) ( − 3 ) ( 5 ) ( 5 ) ( 6 ) ( − 3 ) ( 6 ) ( 5 ) ] = [ 4 − 3 − 2 ] [ 2 − 7 ] + [ − 1 5 6 ] [ − 3 5 ] . \begin{align*}
& =
\small
\begin{bmatrix}
(4)(2) & (4)(-7) \\
(-3)(2) & (-3)(-7) \\
(-2)(2) & (-2)(-7)
\end{bmatrix} +
\begin{bmatrix}
(-1)(-3) & (-1)(5) \\
(5)(-3) & (5)(5) \\
(6)(-3) & (6)(5)
\end{bmatrix}\\[2mm]
& =
\small
\begin{bmatrix}
4 \\ -3 \\ -2
\end{bmatrix}
\begin{bmatrix}
2 & -7
\end{bmatrix} \: + \:
\begin{bmatrix}
-1 \\ 5 \\ 6
\end{bmatrix}
\begin{bmatrix}
-3 & 5
\end{bmatrix}.
\end{align*} = [ ( 4 ) ( 2 ) ( − 3 ) ( 2 ) ( − 2 ) ( 2 ) ( 4 ) ( − 7 ) ( − 3 ) ( − 7 ) ( − 2 ) ( − 7 ) ] + [ ( − 1 ) ( − 3 ) ( 5 ) ( − 3 ) ( 6 ) ( − 3 ) ( − 1 ) ( 5 ) ( 5 ) ( 5 ) ( 6 ) ( 5 ) ] = [ 4 − 3 − 2 ] [ 2 − 7 ] + [ − 1 5 6 ] [ − 3 5 ] . Note that the vectors here are columns of the left-hand matrix and rows of the right-hand matrix. The matrix product is defined only if there are equal numbers of these.
It is not hard to derive the following generalization of Example 2.4.1 to all matrix products.
Theorem 2.4.1 (Matrix multiplication by outer products)
Write the columns of A \mathbf{A} A as a 1 , … , a n \mathbf{a}_1,\dots,\mathbf{a}_n a 1 , … , a n and the rows of B \mathbf{B} B as b 1 T , … , b n T \mathbf{b}_1^T,\dots,\mathbf{b}_n^T b 1 T , … , b n T . Then
A B = ∑ k = 1 n a k b k T . \mathbf{A}\mathbf{B} = \sum_{k=1}^n \mathbf{a}_k \mathbf{b}_k^T. AB = k = 1 ∑ n a k b k T . 2.4.2 Triangular product ¶ Equation (2.4.4) has some interesting structure for the product L U \mathbf{L}\mathbf{U} LU , where L \mathbf{L} L is n × n n\times n n × n and lower triangular (i.e., zero above the main diagonal) and U \mathbf{U} U is n × n n\times n n × n and upper triangular (zero below the diagonal).
Example 2.4.2 (Triangular outer products)
Example 2.4.2 We explore the outer product formula for two random triangular matrices.
L = tril( rand(1:9, 3, 3) )
3×3 Matrix{Int64}:
5 0 0
3 6 0
8 7 8
U = triu( rand(1:9, 3, 3) )
3×3 Matrix{Int64}:
4 3 1
0 7 6
0 0 7
Here are the three outer products in the sum in (2.4.4) :
Tip
Although U[1,:]
is a row of U
, it is a vector, and as such it has a default column interpretation.
3×3 Matrix{Int64}:
20 15 5
12 9 3
32 24 8
3×3 Matrix{Int64}:
0 0 0
0 42 36
0 49 42
3×3 Matrix{Int64}:
0 0 0
0 0 0
0 0 56
Simply because of the triangular zero structures, only the first outer product contributes to the first row and first column of the entire product.
Example 2.4.2 We explore the outer product formula for two random triangular matrices.
L = tril( randi(9, 3, 3) )
U = triu( randi(9, 3, 3) )
Here are the three outer products in the sum in (2.4.4) :
Simply because of the triangular zero structures, only the first outer product contributes to the first row and first column of the entire product.
Example 2.4.2 We explore the outer product formula for two random triangular matrices.
from numpy.random import randint
L = tril(randint(1, 10, size=(3, 3)))
print(L)
[[9 0 0]
[4 5 0]
[5 7 2]]
U = triu(randint(1, 10, size=(3, 3)))
print(U)
[[9 5 5]
[0 7 1]
[0 0 1]]
Here are the three outer products appearing in the sum in (2.4.4) :
print(outer(L[:, 0], U[0, :]))
[[81 45 45]
[36 20 20]
[45 25 25]]
print(outer(L[:, 1], U[1, :]))
[[ 0 0 0]
[ 0 35 5]
[ 0 49 7]]
print(outer(L[:, 2], U[2, :]))
[[0 0 0]
[0 0 0]
[0 0 2]]
Simply because of the triangular zero structures, only the first outer product contributes to the first row and first column of the entire product.
Let the columns of L \mathbf{L} L be written as ℓ k \boldsymbol{\ell}_k ℓ k and the rows of U \mathbf{U} U be written as u k T \mathbf{u}_k^T u k T . Then the first row of L U \mathbf{L}\mathbf{U} LU is
e 1 T ∑ k = 1 n ℓ k u k T = ∑ k = 1 n ( e 1 T ℓ k ) u k T = L 11 u 1 T . \mathbf{e}_1^T \sum_{k=1}^n \boldsymbol{ℓ}_k \mathbf{u}_k^T = \sum_{k=1}^n (\mathbf{e}_1^T \boldsymbol{\ell}_k) \mathbf{u}_k^T = L_{11} \mathbf{u}_1^T. e 1 T k = 1 ∑ n ℓ k u k T = k = 1 ∑ n ( e 1 T ℓ k ) u k T = L 11 u 1 T . Likewise, the first column of L U \mathbf{L}\mathbf{U} LU is
( ∑ k = 1 n ℓ k u k T ) e 1 = ∑ k = 1 n ℓ k ( u k T e 1 ) = U 11 ℓ 1 . \left( \sum_{k=1}^n \mathbf{ℓ}_k \mathbf{u}_k^T\right) \mathbf{e}_1 = \sum_{k=1}^n \mathbf{\ell}_k (\mathbf{u}_k^T \mathbf{e}_1) = U_{11}\boldsymbol{\ell}_1. ( k = 1 ∑ n ℓ k u k T ) e 1 = k = 1 ∑ n ℓ k ( u k T e 1 ) = U 11 ℓ 1 . These two calculations are enough to derive one of the most important algorithms in scientific computing.
2.4.3 Triangular factorization ¶ Our goal is to factor a given n × n n\times n n × n matrix A \mathbf{A} A as the triangular product A = L U \mathbf{A}=\mathbf{L}\mathbf{U} A = LU . It turns out that we have n 2 + n n^2+n n 2 + n total nonzero unknowns in the two triangular matrices, so we set L 11 = ⋯ = L n n = 1 L_{11}=\cdots = L_{nn}=1 L 11 = ⋯ = L nn = 1 , making L \mathbf{L} L a unit lower triangular matrix.
Example 2.4.3 (LU factorization)
Example 2.4.3 For illustration, we work on a 4 × 4 4 \times 4 4 × 4 matrix. We name it with a subscript in preparation for what comes.
A₁ = [
2 0 4 3
-4 5 -7 -10
1 15 2 -4.5
-2 0 2 -13
];
L = diagm(ones(4))
U = zeros(4, 4);
Now we appeal to (2.4.5) . Since L 11 = 1 L_{11}=1 L 11 = 1 , we see that the first row of U \mathbf{U} U is just the first row of A 1 \mathbf{A}_1 A 1 .
4×4 Matrix{Float64}:
2.0 0.0 4.0 3.0
0.0 0.0 0.0 0.0
0.0 0.0 0.0 0.0
0.0 0.0 0.0 0.0
From (2.4.6) , we see that we can find the first column of L \mathbf{L} L from the first column of A 1 \mathbf{A}_1 A 1 .
L[:, 1] = A₁[:, 1] / U[1, 1]
L
4×4 Matrix{Float64}:
1.0 0.0 0.0 0.0
-2.0 1.0 0.0 0.0
0.5 0.0 1.0 0.0
-1.0 0.0 0.0 1.0
We have obtained the first term in the sum (2.4.4) for L U \mathbf{L}\mathbf{U} LU , and we subtract it away from A 1 \mathbf{A}_1 A 1 .
A₂ = A₁ - L[:, 1] * U[1, :]'
4×4 Matrix{Float64}:
0.0 0.0 0.0 0.0
0.0 5.0 1.0 -4.0
0.0 15.0 0.0 -6.0
0.0 0.0 6.0 -10.0
Now A 2 = ℓ 2 u 2 T + ℓ 3 u 3 T + ℓ 4 u 4 T . \mathbf{A}_2 = \boldsymbol{\ell}_2\mathbf{u}_2^T + \boldsymbol{\ell}_3\mathbf{u}_3^T + \boldsymbol{\ell}_4\mathbf{u}_4^T. A 2 = ℓ 2 u 2 T + ℓ 3 u 3 T + ℓ 4 u 4 T . If we ignore the first row and first column of the matrices in this equation, then in what remains we are in the same situation as at the start. Specifically, only ℓ 2 u 2 T \boldsymbol{\ell}_2\mathbf{u}_2^T ℓ 2 u 2 T has any effect on the second row and column, so we can deduce them now.
U[2, :] = A₂[2, :]
L[:, 2] = A₂[:, 2] / U[2, 2]
L
4×4 Matrix{Float64}:
1.0 0.0 0.0 0.0
-2.0 1.0 0.0 0.0
0.5 3.0 1.0 0.0
-1.0 0.0 0.0 1.0
If we subtract off the latest outer product, we have a matrix that is zero in the first two rows and columns.
A₃ = A₂ - L[:, 2] * U[2, :]'
4×4 Matrix{Float64}:
0.0 0.0 0.0 0.0
0.0 0.0 0.0 0.0
0.0 0.0 -3.0 6.0
0.0 0.0 6.0 -10.0
Now we can deal with the lower right 2 × 2 2\times 2 2 × 2 submatrix of the remainder in a similar fashion.
U[3, :] = A₃[3, :]
L[:, 3] = A₃[:, 3] / U[3, 3]
A₄ = A₃ - L[:, 3] * U[3, :]'
4×4 Matrix{Float64}:
0.0 0.0 0.0 0.0
0.0 0.0 0.0 0.0
0.0 0.0 0.0 0.0
0.0 0.0 0.0 2.0
Finally, we pick up the last unknown in the factors.
We now have all of L \mathbf{L} L ,
4×4 Matrix{Float64}:
1.0 0.0 -0.0 0.0
-2.0 1.0 -0.0 0.0
0.5 3.0 1.0 0.0
-1.0 0.0 -2.0 1.0
and all of U \mathbf{U} U ,
4×4 Matrix{Float64}:
2.0 0.0 4.0 3.0
0.0 5.0 1.0 -4.0
0.0 0.0 -3.0 6.0
0.0 0.0 0.0 2.0
We can verify that we have a correct factorization of the original matrix by computing the backward error:
4×4 Matrix{Float64}:
0.0 0.0 0.0 0.0
0.0 0.0 0.0 0.0
0.0 0.0 0.0 0.0
0.0 0.0 0.0 0.0
IIn floating point, we cannot expect the difference to be exactly zero as we found in this toy example. Instead, we would be satisfied to see that each element of the difference above is comparable in size to machine precision.
Example 2.4.3 For illustration, we work on a 4 × 4 4 \times 4 4 × 4 matrix. We name it with a subscript in preparation for what comes.
A_1 = [
2 0 4 3
-4 5 -7 -10
1 15 2 -4.5
-2 0 2 -13
];
L = eye(4);
U = zeros(4, 4);
Now we appeal to (2.4.5) . Since L 11 = 1 L_{11}=1 L 11 = 1 , we see that the first row of U \mathbf{U} U is just the first row of A 1 \mathbf{A}_1 A 1 .
From (2.4.6) , we see that we can find the first column of L \mathbf{L} L from the first column of A 1 \mathbf{A}_1 A 1 .
L(:, 1) = A_1(:, 1) / U(1, 1)
We have obtained the first term in the sum (2.4.4) for L U \mathbf{L}\mathbf{U} LU , and we subtract it away from A 1 \mathbf{A}_1 A 1 .
A_2 = A_1 - L(:, 1) * U(1, :)
Now A 2 = ℓ 2 u 2 T + ℓ 3 u 3 T + ℓ 4 u 4 T . \mathbf{A}_2 = \boldsymbol{\ell}_2\mathbf{u}_2^T + \boldsymbol{\ell}_3\mathbf{u}_3^T + \boldsymbol{\ell}_4\mathbf{u}_4^T. A 2 = ℓ 2 u 2 T + ℓ 3 u 3 T + ℓ 4 u 4 T . If we ignore the first row and first column of the matrices in this equation, then in what remains we are in the same situation as at the start. Specifically, only ℓ 2 u 2 T \boldsymbol{\ell}_2\mathbf{u}_2^T ℓ 2 u 2 T has any effect on the second row and column, so we can deduce them now.
U(2, :) = A_2(2, :)
L(:, 2) = A_2(:, 2) / U(2, 2)
If we subtract off the latest outer product, we have a matrix that is zero in the first two rows and columns.
A_3 = A_2 - L(:, 2) * U(2, :)
Now we can deal with the lower right 2 × 2 2\times 2 2 × 2 submatrix of the remainder in a similar fashion.
U(3, :) = A_3(3, :);
L(:, 3) = A_3(:, 3) / U(3, 3);
A_4 = A_3 - L(:, 3) * U(3, :)
Finally, we pick up the last unknown in the factors.
We now have all of L \mathbf{L} L ,
and all of U \mathbf{U} U ,
We can verify that we have a correct factorization of the original matrix by computing the backward error:
In floating point, we cannot expect the difference to be exactly zero as we found in this toy example. Instead, we would be satisfied to see that each element of the difference above is comparable in size to machine precision.
Example 2.4.3 For illustration, we work on a 4 × 4 4 \times 4 4 × 4 matrix. We name it with a subscript in preparation for what comes.
A_1 = array([
[2, 0, 4, 3],
[-4, 5, -7, -10],
[1, 15, 2, -4.5],
[-2, 0, 2, -13]
])
L = eye(4)
U = zeros((4, 4));
Now we appeal to (2.4.5) . Since L 11 = 1 L_{11}=1 L 11 = 1 , we see that the first row of U \mathbf{U} U is just the first row of A 1 \mathbf{A}_1 A 1 .
U[0, :] = A_1[0, :]
print(U)
[[2. 0. 4. 3.]
[0. 0. 0. 0.]
[0. 0. 0. 0.]
[0. 0. 0. 0.]]
From (2.4.6) , we see that we can find the first column of L \mathbf{L} L from the first column of A 1 \mathbf{A}_1 A 1 .
L[:, 0] = A_1[:, 0] / U[0, 0]
print(L)
[[ 1. 0. 0. 0. ]
[-2. 1. 0. 0. ]
[ 0.5 0. 1. 0. ]
[-1. 0. 0. 1. ]]
We have obtained the first term in the sum (2.4.4) for L U \mathbf{L}\mathbf{U} LU , and we subtract it away from A 1 \mathbf{A}_1 A 1 .
A_2 = A_1 - outer(L[:, 0], U[0, :])
Now A 2 = ℓ 2 u 2 T + ℓ 3 u 3 T + ℓ 4 u 4 T . \mathbf{A}_2 = \boldsymbol{\ell}_2\mathbf{u}_2^T + \boldsymbol{\ell}_3\mathbf{u}_3^T + \boldsymbol{\ell}_4\mathbf{u}_4^T. A 2 = ℓ 2 u 2 T + ℓ 3 u 3 T + ℓ 4 u 4 T . If we ignore the first row and first column of the matrices in this equation, then in what remains we are in the same situation as at the start. Specifically, only ℓ 2 u 2 T \boldsymbol{\ell}_2\mathbf{u}_2^T ℓ 2 u 2 T has any effect on the second row and column, so we can deduce them now.
U[1, :] = A_2[1, :]
L[:, 1] = A_2[:, 1] / U[1, 1]
print(L)
[[ 1. 0. 0. 0. ]
[-2. 1. 0. 0. ]
[ 0.5 3. 1. 0. ]
[-1. 0. 0. 1. ]]
If we subtract off the latest outer product, we have a matrix that is zero in the first two rows and columns.
A_3 = A_2 - outer(L[:, 1], U[1, :])
Now we can deal with the lower right 2 × 2 2\times 2 2 × 2 submatrix of the remainder in a similar fashion.
U[2, :] = A_3[2, :]
L[:, 2] = A_3[:, 2] / U[2, 2]
A_4 = A_3 - outer(L[:, 2], U[2, :])
Finally, we pick up the last unknown in the factors.
We now have all of L \mathbf{L} L ,
[[ 1. 0. -0. 0. ]
[-2. 1. -0. 0. ]
[ 0.5 3. 1. 0. ]
[-1. 0. -2. 1. ]]
and all of U \mathbf{U} U ,
[[ 2. 0. 4. 3.]
[ 0. 5. 1. -4.]
[ 0. 0. -3. 6.]
[ 0. 0. 0. 2.]]
We can verify that we have a correct factorization of the original matrix by computing the backward error:
array([[0., 0., 0., 0.],
[0., 0., 0., 0.],
[0., 0., 0., 0.],
[0., 0., 0., 0.]])
In floating point, we cannot expect the difference to be exactly zero as we found in this toy example. Instead, we would be satisfied to see that each element of the difference above is comparable in size to machine precision.
We have arrived at the linchpin of solving linear systems.
Definition 2.4.1 (LU factorization)
Given n × n n\times n n × n matrix A \mathbf{A} A , its LU factorization is
A = L U , \mathbf{A} = \mathbf{L}\mathbf{U}, A = LU , where L \mathbf{L} L is a unit lower triangular matrix and U \mathbf{U} U is an upper triangular matrix.
The outer product algorithm for LU factorization seen in Demo 2.4.3 is coded as Function 2.4.1 .
LU factorization (not stable)1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
"""
lufact(A)
Compute the LU factorization of square matrix `A`, returning the
factors.
"""
function lufact(A)
n = size(A, 1) # detect the dimensions from the input
L = diagm(ones(n)) # ones on main diagonal, zeros elsewhere
U = zeros(n, n)
Aₖ = float(copy(A)) # make a working copy
# Reduction by outer products
for k in 1:n-1
U[k, :] = Aₖ[k, :]
L[:, k] = Aₖ[:, k] / U[k, k]
Aₖ -= L[:, k] * U[k, :]'
end
U[n, n] = Aₖ[n, n]
return LowerTriangular(L), UpperTriangular(U)
end
About the code
Line 11 of Function 2.4.1 points out two subtle Julia issues. First, vectors and matrix variables are really just references to blocks of memory. Such a reference is much more efficient to pass around than the complete contents of the array. However, it means that a statement Aₖ=A
just clones the array reference of A
into the variable Aₖ
. Any changes made to entries of Aₖ
would then also be made to entries of A
because they refer to the same location in memory. In this context we don’t want to change the original matrix, so we use copy
here to create an independent copy of the array contents and a new reference to them.
The second issue is that even when A
has all integer entries, its LU factors may not. So we convert Aₖ
to floating point so that line 17 will not fail due to the creation of floating-point values in an integer matrix. An alternative would be to require the caller to provide a floating-point array in the first place.
LU factorization (not stable)1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
"""
lufact(A)
Compute the LU factorization of square matrix `A`, returning the
factors.
"""
function lufact(A)
n = size(A, 1) # detect the dimensions from the input
L = diagm(ones(n)) # ones on main diagonal, zeros elsewhere
U = zeros(n, n)
Aₖ = float(copy(A)) # make a working copy
# Reduction by outer products
for k in 1:n-1
U[k, :] = Aₖ[k, :]
L[:, k] = Aₖ[:, k] / U[k, k]
Aₖ -= L[:, k] * U[k, :]'
end
U[n, n] = Aₖ[n, n]
return LowerTriangular(L), UpperTriangular(U)
end
About the code
Line 11 of Function 2.4.1 points out two subtle Julia issues. First, vectors and matrix variables are really just references to blocks of memory. Such a reference is much more efficient to pass around than the complete contents of the array. However, it means that a statement Aₖ=A
just clones the array reference of A
into the variable Aₖ
. Any changes made to entries of Aₖ
would then also be made to entries of A
because they refer to the same location in memory. In this context we don’t want to change the original matrix, so we use copy
here to create an independent copy of the array contents and a new reference to them.
The second issue is that even when A
has all integer entries, its LU factors may not. So we convert Aₖ
to floating point so that line 17 will not fail due to the creation of floating-point values in an integer matrix. An alternative would be to require the caller to provide a floating-point array in the first place.
LU factorization (not stable)1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
def lufact(A):
"""
lufact(A)
Compute the LU factorization of square matrix A, returning the
factors.
"""
n = A.shape[0] # detect the dimensions from the input
L = np.eye(n) # ones on main diagonal, np.zeros elsewhere
U = np.zeros((n, n))
A_k = np.copy(A) # make a working np.copy
# Reduction by np.outer products
for k in range(n-1):
U[k, :] = A_k[k, :]
L[:, k] = A_k[:, k] / U[k,k]
A_k -= np.outer(L[:,k], U[k,:])
U[n-1, n-1] = A_k[n-1, n-1]
return L, U
About the code
Line 11 of Function 2.4.1 points out a subtle issue. Array variables are really just references to blocks of memory. Such a reference is much more efficient to pass around than the complete contents of the array. However, it means that a statement A_k = A
would just clone the array reference of A
into the new variable. Any changes made to entries of A_k
would then also be made to entries of A
, because they refer to the same location in memory. In this context, we don’t want to change the original matrix, so we use copy
here to create an independent copy of the array contents and a new reference to them.
2.4.4 Gaussian elimination and linear systems ¶ In your first matrix algebra course, you probably learned a triangularization technique called Gaussian elimination or row elimination to solve a linear system A x = b \mathbf{A}\mathbf{x}=\mathbf{b} Ax = b . In most presentations, you form an augmented matrix [ A b ] [\mathbf{A}\;\mathbf{b}] [ A b ] and do row operations until the system reaches an upper triangular form, followed by backward substitution. LU factorization is equivalent to Gaussian elimination in which no row swaps are performed, and the elimination procedure produces the factors if you keep track of the row multipliers appropriately.
Like Gaussian elimination, the primary use of LU factorization is to solve a linear system. It reduces a given linear system to two triangular ones. From this, solving A x = b \mathbf{A}\mathbf{x}=\mathbf{b} Ax = b follows immediately from associativity:
b = A x = ( L U ) x = L ( U x ) . \mathbf{b} = \mathbf{A} \mathbf{x} = (\mathbf{L} \mathbf{U}) \mathbf{x} = \mathbf{L} (\mathbf{U} \mathbf{x}). b = Ax = ( LU ) x = L ( Ux ) . Defining z = U x \mathbf{z} = \mathbf{U} \mathbf{x} z = Ux leads to the following.
Algorithm 2.4.2 (Solution of linear systems by LU factorization (unstable))
Factor L U = A \mathbf{L}\mathbf{U}=\mathbf{A} LU = A . Solve L z = b \mathbf{L}\mathbf{z}=\mathbf{b} Lz = b for z \mathbf{z} z using forward substitution. Solve U x = z \mathbf{U}\mathbf{x}=\mathbf{z} Ux = z for x \mathbf{x} x using backward substitution. A key advantage of the factorization point of view is that it depends only on the matrix A \mathbf{A} A . If systems are to be solved for a single A \mathbf{A} A but multiple different versions of b \mathbf{b} b , then the factorization approach is more efficient, as we’ll see in Efficiency of matrix computations .
Example 2.4.4 (Solving a linear system by LU factors)
Example 2.4.4 Here are the data for a linear system A x = b \mathbf{A}\mathbf{x}=\mathbf{b} Ax = b .
A = [2 0 4 3; -4 5 -7 -10; 1 15 2 -4.5; -2 0 2 -13];
b = [4,9,9,4];
We apply Function 2.4.1 and then do two triangular solves.
L, U = FNC.lufact(A)
z = FNC.forwardsub(L, b)
x = FNC.backsub(U, z)
4-element Vector{Float64}:
192.66666666666666
-15.533333333333335
-65.33333333333333
-40.0
A check on the residual assures us that we found the solution.
4-element Vector{Float64}:
0.0
-5.684341886080802e-14
2.842170943040401e-14
0.0
Example 2.4.4 Here are the data for a linear system A x = b \mathbf{A}\mathbf{x}=\mathbf{b} Ax = b .
A = [2 0 4 3; -4 5 -7 -10; 1 15 2 -4.5; -2 0 2 -13];
b = [4; 9; 9; 4];
We apply Function 2.4.1 and then do two triangular solves.
[L, U] = lufact(A)
z = forwardsub(L, b);
x = backsub(U, z);
A check on the residual assures us that we found the solution.
Example 2.4.4 Here are the data for a linear system A x = b \mathbf{A}\mathbf{x}=\mathbf{b} Ax = b .
A = array([
[2, 0, 4, 3],
[-4, 5, -7, -10],
[1, 15, 2, -4.5],
[-2, 0, 2, -13]
])
b = array([4, 9, 9, 4])
We apply Function 2.4.1 and then do two triangular solves.
L, U = FNC.lufact(A)
z = FNC.forwardsub(L, b)
x = FNC.backsub(U, z)
A check on the residual assures us that we found the solution.
array([0.00000000e+00, 0.00000000e+00, 2.84217094e-14, 0.00000000e+00])
As noted in the descriptions of Function 2.4.1 and Algorithm 2.4.2 , the LU factorization as we have seen it so far is not stable for all matrices. In fact, it does not always even exist. The missing element is the row swapping allowed in Gaussian elimination. We will address these issues in Row pivoting .
2.4.5 Exercises ¶ ✍ For each matrix, produce an LU factorization by hand.
(a) [ 2 3 4 4 5 10 4 8 2 ] \quad \displaystyle \begin{bmatrix}
2 & 3 & 4 \\
4 & 5 & 10 \\
4 & 8 & 2
\end{bmatrix}\qquad ⎣ ⎡ 2 4 4 3 5 8 4 10 2 ⎦ ⎤
(b) [ 6 − 2 − 4 4 3 − 3 − 6 1 − 12 8 21 − 8 − 6 0 − 10 7 ] \quad \displaystyle \begin{bmatrix}
6 & -2 & -4 & 4\\
3 & -3 & -6 & 1 \\
-12 & 8 & 21 & -8 \\
-6 & 0 & -10 & 7
\end{bmatrix} ⎣ ⎡ 6 3 − 12 − 6 − 2 − 3 8 0 − 4 − 6 21 − 10 4 1 − 8 7 ⎦ ⎤
⌨ The matrices
T ( x , y ) = [ 1 0 0 0 1 0 x y 1 ] , R ( θ ) = [ cos θ sin θ 0 − sin θ cos θ 0 0 0 1 ] \mathbf{T}(x,y) = \begin{bmatrix}
1 & 0 & 0 \\ 0 & 1 & 0 \\ x & y & 1
\end{bmatrix},\qquad
\mathbf{R}(\theta) = \begin{bmatrix}
\cos\theta & \sin \theta & 0 \\ -\sin\theta & \cos \theta & 0 \\ 0 & 0 & 1
\end{bmatrix} T ( x , y ) = ⎣ ⎡ 1 0 x 0 1 y 0 0 1 ⎦ ⎤ , R ( θ ) = ⎣ ⎡ cos θ − sin θ 0 sin θ cos θ 0 0 0 1 ⎦ ⎤ are used to represent translations and rotations of plane points in computer graphics. For the following, let
A = T ( 3 , − 1 ) R ( π / 5 ) T ( − 3 , 1 ) , z = [ 2 2 1 ] . \mathbf{A} = \mathbf{T}(3,-1)\mathbf{R}(\pi/5)\mathbf{T}(-3,1), \qquad \mathbf{z} = \begin{bmatrix}
2 \\ 2 \\ 1
\end{bmatrix}. A = T ( 3 , − 1 ) R ( π /5 ) T ( − 3 , 1 ) , z = ⎣ ⎡ 2 2 1 ⎦ ⎤ . (a) Find b = A z \mathbf{b} = \mathbf{A}\mathbf{z} b = Az .
(b) Use Function 2.4.1 to find the LU factorization of A \mathbf{A} A .
(c) Use the factors with triangular substitutions in order to solve A x = b \mathbf{A}\mathbf{x}=\mathbf{b} Ax = b , and find x − z \mathbf{x}-\mathbf{z} x − z .
⌨ Define
A = [ 1 0 0 0 1 0 12 1 1 0 0 0 0 1 1 0 0 0 0 1 1 0 0 0 0 1 0 ] , x ^ = [ 0 1 / 3 2 / 3 1 4 / 3 ] , b = A x ^ . \mathbf{A}= \begin{bmatrix}
1 & 0 & 0 & 0 & 10^{12} \\
1 & 1 & 0 & 0 & 0 \\
0 & 1 & 1 & 0 & 0 \\
0 & 0 & 1 & 1 & 0 \\
0 & 0 & 0 & 1 & 0
\end{bmatrix},
\quad \hat{\mathbf{x}} = \begin{bmatrix}
0 \\ 1/3 \\ 2/3 \\ 1 \\ 4/3
\end{bmatrix},
\quad \mathbf{b} = \mathbf{A}\hat{\mathbf{x}}. A = ⎣ ⎡ 1 1 0 0 0 0 1 1 0 0 0 0 1 1 0 0 0 0 1 1 1 0 12 0 0 0 0 ⎦ ⎤ , x ^ = ⎣ ⎡ 0 1/3 2/3 1 4/3 ⎦ ⎤ , b = A x ^ . (a) Using Function 2.4.1 and triangular substitutions, solve the linear system A x = b \mathbf{A}\mathbf{x}=\mathbf{b} Ax = b , showing the result. To the nearest integer, how many accurate digits are in the result? (The answer is much less than the full 16 of double precision.)
(b) Repeat part (a) with 1020 as the element in the upper right corner. (The result is even less accurate. We will study the causes of such low accuracy in Conditioning of linear systems .)
⌨ Let
A = [ 1 1 0 1 0 0 0 1 1 0 1 0 0 0 1 1 0 1 1 0 0 1 1 0 1 1 0 0 1 1 0 1 1 0 0 1 ] . \mathbf{A} =
\begin{bmatrix}
1 & 1 & 0 & 1 & 0 & 0 \\
0 & 1 & 1 & 0 & 1 & 0 \\
0 & 0 & 1 & 1 & 0 & 1 \\
1 & 0 & 0 & 1 & 1 & 0 \\
1 & 1 & 0 & 0 & 1 & 1 \\
0 & 1 & 1 & 0 & 0 & 1
\end{bmatrix}. A = ⎣ ⎡ 1 0 0 1 1 0 1 1 0 0 1 1 0 1 1 0 0 1 1 0 1 1 0 0 0 1 0 1 1 0 0 0 1 0 1 1 ⎦ ⎤ . Verify computationally that if A = L U \mathbf{A}=\mathbf{L}\mathbf{U} A = LU is the LU factorization, then the elements of L \mathbf{L} L , U \mathbf{U} U , L − 1 \mathbf{L}^{-1} L − 1 , and U − 1 \mathbf{U}^{-1} U − 1 are all integers. Do not rely just on visual inspection of the numbers; perform a more definitive test.
⌨ Function 2.4.1 factors A = L U \mathbf{A}=\mathbf{L}\mathbf{U} A = LU in such a way that L \mathbf{L} L is a unit lower triangular matrix—that is, has all ones on the diagonal. It is also possible to define the factorization so that U \mathbf{U} U is a unit upper triangular matrix instead. Write a function lufact2
that uses Function 2.4.1 without modification to produce this version of the factorization. (Hint: Begin with the standard LU factorization of A T \mathbf{A}^T A T .) Demonstrate on a nontrivial 4 × 4 4\times 4 4 × 4 example.
When computing the determinant of a matrix by hand, it’s common to use cofactor expansion and apply the definition recursively. But this is terribly inefficient as a function of the matrix size.
(a) ✍ Explain using determinant properties why, if A = L U \mathbf{A}=\mathbf{L}\mathbf{U} A = LU is an LU factorization,
det ( A ) = U 11 U 22 ⋯ U n n = ∏ i = 1 n U i i . \det(\mathbf{A}) = U_{11}U_{22}\cdots U_{nn}=\prod_{i=1}^n U_{ii}. det ( A ) = U 11 U 22 ⋯ U nn = i = 1 ∏ n U ii . (b) ⌨ Using the result of part (a), write a function determinant(A)
that computes the determinant using Function 2.4.1 . Test your function on at least two nontriangular 5 × 5 5\times 5 5 × 5 matrices, comparing your result to the result of the standard det
function.