QR Call
produces the QR decomposition of a
matrix by Householder transformations
- CALL QR(
,
, piv, lindep,
, ord<,
b>);
The QR subroutine returns the following values:
data:image/s3,"s3://crabby-images/47ef3/47ef3ec60ff3aaf917ed6841ebc7b6aa938f509f" alt="q"
- specifies an orthogonal matrix
that is the product of
the Householder transformations applied to the
matrix
, if the
argument is not specified.
In this case, the
Householder transformations
are applied, and
is an
matrix.
If the
argument is specified,
is the
matrix
that has the transposed
Householder transformations
applied
on the
columns of the argument matrix
.
data:image/s3,"s3://crabby-images/c6eb5/c6eb562af786b889124a60152b171a8f19477a4a" alt="r"
- specifies a
upper triangular
matrix
that is the upper part of the
upper triangular matrix
of the QR decomposition of the matrix
.
The matrix
of the QR decomposition
can be obtained by vertical concatenation (by using the
IML operator //) of the
zero matrix to the result matrix
.
- piv
- specifies an
vector of permutations of the
columns of
; that is, on return, the QR decomposition
is computed, not of
, but of the permuted matrix
whose columns are
.
The vector piv corresponds to an
permutation matrix
.
- lindep
- is the number of linearly dependent columns in matrix
detected by applying the
Householder transformations
in the order specified by the argument vector piv.
The inputs to the QR subroutine are as follows:
data:image/s3,"s3://crabby-images/3c86a/3c86a963e2b669a85fee226386935dc6307beb4a" alt="a"
- specifies an
matrix
that is to be
decomposed into the product of the orthogonal matrix
and the upper triangular matrix
.
- ord
- specifies an optional
vector that
specifies the order of Householder transformations
applied to matrix
, as follows:
- ord
![[j{]} \gt 0](images/langref_langrefeq969.gif)
- Column
of
is an initial column, meaning it has to
be processed at the start in increasing order of ord
.
- ord
![[j{]} = 0](images/langref_langrefeq971.gif)
- Column
of
can be permuted in order
of decreasing residual Euclidean norm (pivoting).
- ord
![[j{]} \lt 0](images/langref_langrefeq972.gif)
- Column
of
is a final column, meaning it has to
be processed at the end in decreasing order of ord
.
The default is ord
, in which case the
Householder transformations are done in the same order that
the columns are stored in matrix
(without pivoting).
data:image/s3,"s3://crabby-images/73805/738050eae4c3abc2868bdfb668a197387ffe14f5" alt="b"
- specifies an optional
matrix
that is to be
multiplied by the transposed
matrix
.
If
is specified, the result
contains
the
matrix
.
If
is not specified, the result
contains the
matrix
.
The QR subroutine decomposes an
data:image/s3,"s3://crabby-images/c495f/c495fb0a9a8c2f175cd1ec31a7eb6ccf373dd957" alt="m x n"
matrix
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
into
the product of an
data:image/s3,"s3://crabby-images/8ea52/8ea52988fc9eb9a692ee36f0a2bc715861078553" alt="m x m"
orthogonal matrix
data:image/s3,"s3://crabby-images/0bb7b/0bb7b7fb6e0618ad3417246aeeca1422fe139899" alt="{q}"
and an
data:image/s3,"s3://crabby-images/c495f/c495fb0a9a8c2f175cd1ec31a7eb6ccf373dd957" alt="m x n"
upper triangular matrix
data:image/s3,"s3://crabby-images/0a2cf/0a2cfb941ddaa2d2a7e6b6d9601878001f69d131" alt="\widetilde{{{r}}}"
, so that
data:image/s3,"s3://crabby-images/f54c9/f54c9dde009d3772f5879684fb8b23245abe509b" alt="{a}{{{{\pi}}}}= {q}\widetilde{{{r}}}, \hspace*{.2in} {q}^' {q}= {q}{q}^' = {i}_m"
by means of
data:image/s3,"s3://crabby-images/82eb3/82eb347ced85de97912e1964469bbc11b80dbd38" alt="\min(m,n)"
Householder transformations.
The
data:image/s3,"s3://crabby-images/8ea52/8ea52988fc9eb9a692ee36f0a2bc715861078553" alt="m x m"
orthogonal matrix
data:image/s3,"s3://crabby-images/0bb7b/0bb7b7fb6e0618ad3417246aeeca1422fe139899" alt="{q}"
is computed only
if the last argument
data:image/s3,"s3://crabby-images/73805/738050eae4c3abc2868bdfb668a197387ffe14f5" alt="b"
is not specified, as in the following code:
call qr(q,r,piv,lindep,a,ord);
In many applications, the number of rows,
data:image/s3,"s3://crabby-images/be1ac/be1acfb5e2ef8c1161442c83f8d9b64b570626ff" alt="m"
, is very large.
In these cases, the explicit computation of the
data:image/s3,"s3://crabby-images/8ea52/8ea52988fc9eb9a692ee36f0a2bc715861078553" alt="m x m"
matrix
data:image/s3,"s3://crabby-images/0bb7b/0bb7b7fb6e0618ad3417246aeeca1422fe139899" alt="{q}"
can require too much memory or time.
In the usual case where
data:image/s3,"s3://crabby-images/515e7/515e752e1b49444036ec7f4ad16d5284d3860038" alt="m \gt n"
,
![{a}= [ * & * & * \ {}* & * & * \ {}* & * & * \ {}* & * & * \ {}* & * & *... ... & 0 & * ] \ {q}= [ {q}_1 {q}_2 ], & & \widetilde{{{r}}} = [ {{r}}\ 0 ]](images/langref_langrefeq975.gif)
where
data:image/s3,"s3://crabby-images/6c306/6c306ac8b358fdb51acfee12aaad5d23d6d0633e" alt="{{r}}"
is the result returned by the QR subroutine.
The
data:image/s3,"s3://crabby-images/a5ae5/a5ae5bd45e56315109d7592bd087485e20f89f3a" alt="n"
columns of matrix
data:image/s3,"s3://crabby-images/57014/570146a8a137a137cca2fc995e591ab4a64afe95" alt="{q}_1"
provide an
orthonormal basis for the
data:image/s3,"s3://crabby-images/a5ae5/a5ae5bd45e56315109d7592bd087485e20f89f3a" alt="n"
columns of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
and are called the
range space of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
.
Since the
data:image/s3,"s3://crabby-images/96d4c/96d4cc2b6b94b4c4cd3fe8ec533b1ddda39c82ec" alt="m-n"
columns of
data:image/s3,"s3://crabby-images/d13d1/d13d140ad993704b6c05b6cc6f821d94f5367f86" alt="{q}_2"
are orthogonal to the
data:image/s3,"s3://crabby-images/a5ae5/a5ae5bd45e56315109d7592bd087485e20f89f3a" alt="n"
columns of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
,
data:image/s3,"s3://crabby-images/ca08a/ca08a4772733ab3acc37e02428be8f3439f5f9d2" alt="{q}_2^' {a}= 0"
, they provide
an orthonormal basis for the orthogonal complement of the
columns of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
and are called the
null space of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
.
In the case where
data:image/s3,"s3://crabby-images/b55bc/b55bc57c042c7608a578ac00292776e56e215ea2" alt="m \lt n"
,
![{a}= [ * & * & * & * & * \ {}* & * & * & * & * \ {}* & * & * & * & * ] & &... ...} = {{r}}= [ * & * & * & * & * \ 0 & * & * & * & * \ 0 & 0 & * & * & * ]](images/langref_langrefeq978.gif)
Specifying the argument
ord as an
data:image/s3,"s3://crabby-images/a5ae5/a5ae5bd45e56315109d7592bd087485e20f89f3a" alt="n"
vector lets
you specify a special order of the columns in matrix
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
on which the Householder transformations are applied.
When you specify the
ord argument, the columns
of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
can be divided into the following groups:
- ord
: Column
of
is an
initial column, meaning it has to be processed
at the start in increasing order of ord
.
This specification defines the first
columns of
that are to be processed.
- ord
: Column
of
is a
pivot column, meaning it is to be processed
in order of decreasing residual Euclidean norms.
The pivot columns of
are processed
after the
initial columns
and before the
final columns.
- ord
: Column
of
is a
final column, meaning it has to be processed
at the end in decreasing order of ord
.
This specification defines the last
columns of
that are to be processed.
If
, some of these columns
will not be processed at all.
There are two special cases:
- If you do not specify the ord argument,
the default values ord
are used.
In this case, Householder transformations are
done in the same order in which the columns
are stored in
(without pivoting).
- If you set all components of ord to zero,
the Householder transformations are done in order of
decreasing Euclidean norms of the columns of
.
The resulting
data:image/s3,"s3://crabby-images/30efe/30efea6cdf6942f07ad57784be8b52e62e1d3cea" alt="n x 1"
vector
piv specifies
the permutation of the columns of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
on which the
Householder transformations are applied; that is,
on return, the QR decomposition is computed, not of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
, but of the matrix with columns that are permuted
in the order
![{a}_{piv[1]}, ... , {a}_{piv[n]}](images/langref_langrefeq981.gif)
.
To check the QR decomposition, use the following statements
to compute the three residual sum of squares, represented by
the variables SS0, SS1, and SS2, which should be close to zero:
m = nrow(a); n = ncol(a);
call qr(q,r,piv,lindep,a,ord);
ss0 = ssq(a[ ,piv] - q[,1:n] * r);
ss1 = ssq(q * q` - i(m));
ss2 = ssq(q` * q - i(m));
If the QR subroutine detects linearly dependent columns
while processing matrix
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
, the column order given in
the result vector
piv can differ from an explicitly
specified order in the argument vector
ord.
If a column of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
is found to be linearly
dependent on columns already processed, this
column is swapped to the end of matrix
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
.
The order of columns in the result matrix
data:image/s3,"s3://crabby-images/6c306/6c306ac8b358fdb51acfee12aaad5d23d6d0633e" alt="{{r}}"
corresponds to the order of columns processed in
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
.
The swapping of a linearly dependent column of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
to the end of the matrix corresponds to the swapping
of the same column in
data:image/s3,"s3://crabby-images/6c306/6c306ac8b358fdb51acfee12aaad5d23d6d0633e" alt="{{r}}"
and leads to a zero row
at the end of the upper triangular matrix
data:image/s3,"s3://crabby-images/6c306/6c306ac8b358fdb51acfee12aaad5d23d6d0633e" alt="{{r}}"
.
The scalar result
lindep counts the number of
linearly dependent columns that are detected in constructing
the first
data:image/s3,"s3://crabby-images/82eb3/82eb347ced85de97912e1964469bbc11b80dbd38" alt="\min(m,n)"
Householder transformations in
the order specified by the argument vector
ord.
The test of linear dependence depends on the size of the
singularity criterion used; currently it is specified as 1E-8.
Solving the linear system
data:image/s3,"s3://crabby-images/61cfd/61cfd4efb8c5bd766d4ea0e1794cda4ad0c036e3" alt="{{r}}{x}= {q}^' {b}"
with an upper triangular matrix
data:image/s3,"s3://crabby-images/6c306/6c306ac8b358fdb51acfee12aaad5d23d6d0633e" alt="{{r}}"
whose columns are
permuted corresponding to the result vector
piv
leads to a solution
data:image/s3,"s3://crabby-images/ada6d/ada6df7ca3d9f0120cdf07fb3777a59d6a2b70bd" alt="{x}"
with permuted components.
You can reorder the components of
data:image/s3,"s3://crabby-images/ada6d/ada6df7ca3d9f0120cdf07fb3777a59d6a2b70bd" alt="{x}"
by using the index vector
piv at the left-hand side of an expression, as follows:
call qr(qtb,r,piv,lindep,a,ord,b);
x[piv] = inv(r) * qtb[1:n,1:p];
The following example solves the full-rank
linear least squares problem.
Specify the argument
data:image/s3,"s3://crabby-images/73805/738050eae4c3abc2868bdfb668a197387ffe14f5" alt="b"
as an
data:image/s3,"s3://crabby-images/657c5/657c51974f3f2a4de7bb7085837d0806738660e0" alt="m x p"
matrix
data:image/s3,"s3://crabby-images/6f4b5/6f4b530783997925bbe674f54b4821090d9675d7" alt="{b}"
, as follows:
call qr(q,r,piv,lindep,a,ord,b);
When you specify the
data:image/s3,"s3://crabby-images/73805/738050eae4c3abc2868bdfb668a197387ffe14f5" alt="b"
argument, the QR call computes the
matrix
data:image/s3,"s3://crabby-images/657f9/657f993f9d9d04f20a0370a9aec2625294403044" alt="{q}^' {b}"
(instead of
data:image/s3,"s3://crabby-images/0bb7b/0bb7b7fb6e0618ad3417246aeeca1422fe139899" alt="{q}"
) as the result
data:image/s3,"s3://crabby-images/47ef3/47ef3ec60ff3aaf917ed6841ebc7b6aa938f509f" alt="q"
.
Now you can compute the
data:image/s3,"s3://crabby-images/ff118/ff11898ef82fe915b329b8409951888f15995bb2" alt="p"
least squares solutions
data:image/s3,"s3://crabby-images/3acfd/3acfd2ec0f34a855c775e06d68aac283a1141dae" alt="{x}_k"
of an overdetermined linear system with
an
data:image/s3,"s3://crabby-images/0424b/0424b0f8300affc58afc861981ac6ace792d9d4b" alt="m x n, m \gt n"
coefficient matrix
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
,
rank(
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
) =
data:image/s3,"s3://crabby-images/a5ae5/a5ae5bd45e56315109d7592bd087485e20f89f3a" alt="n"
, and
data:image/s3,"s3://crabby-images/ff118/ff11898ef82fe915b329b8409951888f15995bb2" alt="p"
right-hand sides
data:image/s3,"s3://crabby-images/24df3/24df358a152f8de2962c021c5edb1204e117af06" alt="{b}_k"
stored as the columns of the
data:image/s3,"s3://crabby-images/657c5/657c51974f3f2a4de7bb7085837d0806738660e0" alt="m x p"
matrix
data:image/s3,"s3://crabby-images/6f4b5/6f4b530783997925bbe674f54b4821090d9675d7" alt="{b}"
:
data:image/s3,"s3://crabby-images/40680/4068007398230e5a5f4c65559258f0b7d85cc6b9" alt="\min_{{x}_k} \vert {a}{x}_k - {b}_k \vert^2, \hspace*{0.2in} k = 1, ... ,p"
where
data:image/s3,"s3://crabby-images/5456c/5456c5330e2ab230b56f2c8146ba7b4ea12b017e" alt="\vert \cdot \vert"
is the Euclidean vector norm.
This is accomplished by solving the
data:image/s3,"s3://crabby-images/ff118/ff11898ef82fe915b329b8409951888f15995bb2" alt="p"
upper
triangular systems with back-substitution:
data:image/s3,"s3://crabby-images/627a7/627a7e4519308ee0b65ab1d7cdcc59c29347a1c3" alt="{x}_k = {{{{\pi}}}}^' {{r}}^{-1}{q}_1^' {b}_k, \hspace*{0.2in} k = 1, ... ,p"
For most applications,
data:image/s3,"s3://crabby-images/be1ac/be1acfb5e2ef8c1161442c83f8d9b64b570626ff" alt="m"
, the number of rows of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
, is much larger than
data:image/s3,"s3://crabby-images/a5ae5/a5ae5bd45e56315109d7592bd087485e20f89f3a" alt="n"
, the number of columns
of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
, or
data:image/s3,"s3://crabby-images/ff118/ff11898ef82fe915b329b8409951888f15995bb2" alt="p"
, the number of right-hand sides.
In these cases, you are advised not to compute the large
data:image/s3,"s3://crabby-images/8ea52/8ea52988fc9eb9a692ee36f0a2bc715861078553" alt="m x m"
matrix
data:image/s3,"s3://crabby-images/0bb7b/0bb7b7fb6e0618ad3417246aeeca1422fe139899" alt="{q}"
(which can consume too much memory
and time) if you can solve your problem by computing only
the smaller
data:image/s3,"s3://crabby-images/657c5/657c51974f3f2a4de7bb7085837d0806738660e0" alt="m x p"
matrix
data:image/s3,"s3://crabby-images/657f9/657f993f9d9d04f20a0370a9aec2625294403044" alt="{q}^' {b}"
implicitly.
For example, use the first five columns
of the
data:image/s3,"s3://crabby-images/55a19/55a19e9908a099ef650b591058dc7bfc3b868caa" alt="6 x 6"
Hilbert matrix
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
, as follows:
a= { 36 -630 3360 -7560 7560 -2772,
-630 14700 -88200 211680 -220500 83160,
3360 -88200 564480 -1411200 1512000 -582120,
-7560 211680 -1411200 3628800 -3969000 1552320,
7560 -220500 1512000 -3969000 4410000 -1746360,
-2772 83160 -582120 1552320 -1746360 698544 };
b= { 463, -13860, 97020, -258720, 291060, -116424};
n = 5; aa = a[,1:n];
call qr(qtb,r,piv,lindep,aa,,b);
if lindep=0 then x=inv(r)*qtb[1:n];
print x;
Note that you are using only the first
data:image/s3,"s3://crabby-images/a5ae5/a5ae5bd45e56315109d7592bd087485e20f89f3a" alt="n"
rows,
data:image/s3,"s3://crabby-images/b86c2/b86c222ab77ed1889f506ceaf265521199c48cd5" alt="{q}^'_1{b}"
, of QTB.
The
IF-THEN statement of the preceding code can be replaced
by the more efficient
TRISOLV function, as follows:
if lindep=0 then x=trisolv(1,r,qtb[1:n],piv);
print x;
Both cases produce the following output:
X
1
0.5
0.3333333
0.25
0.2
For information about solving rank-deficient linear
least squares problems, see the RZLIND call.