RZLIND Call
computes rank deficient linear least squares
solutions, complete orthogonal factorization,
and Moore-Penrose inverses
- CALL RZLIND( lindep, rup, bup,
, sing<,
b>);
The RZLIND subroutine returns the following values:
- lindep
- is a
scalar giving the number of linear
dependencies that are recognized in
(number
of zeroed rows in rup[n,n]).
- rup
- is the
updated
upper triangular
matrix
containing zero rows corresponding to
zero recognized diagonal elements in the original
.
- bup
- is the
matrix
of right-hand
sides that is updated simultaneously with
.
If
is not specified, bup is not accessible.
The inputs to the RZLIND subroutine are as follows:
data:image/s3,"s3://crabby-images/c6eb5/c6eb562af786b889124a60152b171a8f19477a4a" alt="r"
- specifies the
upper triangular matrix
.
Only the upper triangle of
is used; the lower
triangle can contain any information.
- sing
- is an
optional scalar specifying a relative
singularity criterion for the diagonal elements of
.
The diagonal element
is considered zero if
,
where
is the Euclidean
norm of column
of
. If the value provided
for sing is not positive, the default value
sing
is used, where
is
the relative machine precision.
data:image/s3,"s3://crabby-images/73805/738050eae4c3abc2868bdfb668a197387ffe14f5" alt="b"
- specifies the
optional
matrix
of
right-hand sides that have to be updated or downdated
simultaneously with
.
The singularity test used in the RZLIND subroutine is a relative
test using the Euclidean norms of the columns
data:image/s3,"s3://crabby-images/59376/59376c30161d5e65d06cf360247b6cf700ad117c" alt="{r}_i"
of
data:image/s3,"s3://crabby-images/6c306/6c306ac8b358fdb51acfee12aaad5d23d6d0633e" alt="{{r}}"
.
The diagonal element
data:image/s3,"s3://crabby-images/7205e/7205ec1a8fc9b8ae4be0caed36e13a01cf1f569e" alt="r_{ii}"
is considered
as nearly zero (and the
data:image/s3,"s3://crabby-images/253ed/253ed09d7887bbf81bdc0acad9ada960c19291c8" alt="i"
th row is
zeroed out) if the following test is true:
data:image/s3,"s3://crabby-images/a1833/a18335f9ea8eae4efa9d8afd44259ed3b432b476" alt="r_{ii} \leq {sing} \vert {r}_i \vert, { where } \vert {r}_i \vert = \sqrt{{r}_i^' {r}_i}"
Providing an argument
singdata:image/s3,"s3://crabby-images/4a00c/4a00cdc872bba17556ca6519ab7ed101e764fc1a" alt="\leq 0"
is the same
as omitting the argument
sing in the RZLIND call.
In this case, the default is
singdata:image/s3,"s3://crabby-images/e7ac8/e7ac8524273d83de7444821f7459b55546c21328" alt="= 1000\epsilon"
,
where
data:image/s3,"s3://crabby-images/0f90e/0f90e21c567ebaa6b8dae87f174fcbdecbcdceb4" alt="\epsilon"
is the relative machine precision.
If
data:image/s3,"s3://crabby-images/6c306/6c306ac8b358fdb51acfee12aaad5d23d6d0633e" alt="{{r}}"
is computed by the QR decomposition
data:image/s3,"s3://crabby-images/c1743/c17438af3530b18b218a403e792bde78a51761ec" alt="{a}= {q}{{r}}"
, then
the Euclidean norm of column
data:image/s3,"s3://crabby-images/253ed/253ed09d7887bbf81bdc0acad9ada960c19291c8" alt="i"
of
data:image/s3,"s3://crabby-images/6c306/6c306ac8b358fdb51acfee12aaad5d23d6d0633e" alt="{{r}}"
is the same (except for
rounding errors) as the Euclidean norm of column
data:image/s3,"s3://crabby-images/253ed/253ed09d7887bbf81bdc0acad9ada960c19291c8" alt="i"
of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
.
Consider the following possible
application of the RZLIND subroutine.
Assume that you want to compute the upper
triangular Cholesky factor
data:image/s3,"s3://crabby-images/6c306/6c306ac8b358fdb51acfee12aaad5d23d6d0633e" alt="{{r}}"
of the
data:image/s3,"s3://crabby-images/cd674/cd6748948b4fe1ab19310d079b68f035c192840b" alt="n x n"
positive semidefinite matrix
data:image/s3,"s3://crabby-images/db7db/db7db8c1b4c18e2c6f754913728ad5f09fc577f1" alt="{a}^' {a}"
,
data:image/s3,"s3://crabby-images/617bd/617bdc10d8c55307389b9da4f60312a718f58aa5" alt="{a}^' {a}= {{r}}^' {{r}} { where } {a}\in {\cal r}^{m x n}, {\rm rank}({a})=r, r \leq n \leq m"
The Cholesky factor
data:image/s3,"s3://crabby-images/6c306/6c306ac8b358fdb51acfee12aaad5d23d6d0633e" alt="{{r}}"
of a positive
definite matrix
data:image/s3,"s3://crabby-images/db7db/db7db8c1b4c18e2c6f754913728ad5f09fc577f1" alt="{a}^' {a}"
is unique
(with the exception of the sign of its rows).
However, the Cholesky factor of a positive semidefinite (singular)
matrix
data:image/s3,"s3://crabby-images/db7db/db7db8c1b4c18e2c6f754913728ad5f09fc577f1" alt="{a}^' {a}"
can have many different forms.
In the following example,
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
is a
data:image/s3,"s3://crabby-images/d6b2f/d6b2f2e439cedd2fe3f3119fedc56c7b2571e6d7" alt="12 x 8"
matrix with
linearly dependent columns
data:image/s3,"s3://crabby-images/c0d07/c0d07eb0ee21c9a607f600a51d39075c00ef0e7c" alt="{a}_1 = {a}_2 + {a}_3 + {a}_4"
and
data:image/s3,"s3://crabby-images/b5591/b55915a7a2ea9389c8beeba12319b1fb6e7a854a" alt="{a}_1 = {a}_5 + {a}_6 + {a}_7"
with
data:image/s3,"s3://crabby-images/712e1/712e13c6083bdfc899a7e4d829fdd0ef76156187" alt="r=6"
,
data:image/s3,"s3://crabby-images/38253/38253d765345134c91149685f5b76e4b623938f9" alt="n=8"
, and
data:image/s3,"s3://crabby-images/75ebb/75ebb0e4315346819f0129919c597ea79cc357f7" alt="m=12"
.
a = {1 1 0 0 1 0 0,
1 1 0 0 1 0 0,
1 1 0 0 0 1 0,
1 1 0 0 0 0 1,
1 0 1 0 1 0 0,
1 0 1 0 0 1 0,
1 0 1 0 0 1 0,
1 0 1 0 0 0 1,
1 0 0 1 1 0 0,
1 0 0 1 0 1 0,
1 0 0 1 0 0 1,
1 0 0 1 0 0 1};
a = a || uniform(j(12,1,1));
aa = a` * a;
m = nrow(a); n = ncol(a);
Applying the
ROOT function to the coefficient
matrix
data:image/s3,"s3://crabby-images/db7db/db7db8c1b4c18e2c6f754913728ad5f09fc577f1" alt="{a}^' {a}"
of the normal equations generates an
upper triangular matrix
data:image/s3,"s3://crabby-images/b67d3/b67d34d035d19b433c37bce622ddd4c8f9d0de1a" alt="{{r}}_1"
where
linearly dependent rows are zeroed out. You can
verify that
data:image/s3,"s3://crabby-images/44a43/44a43a2f65dc0c054f8c2a51e524fd14e998bc17" alt="{a}^' {a}= {{r}}^'_1 {{r}}_1"
.
Here is the code:
r1 = root(aa);
ss1 = ssq(aa - r1` * r1);
print ss1 r1 [format=best6.];
Applying the QR subroutine with column pivoting on the
original matrix
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
yields a different result, but you
can also verify
data:image/s3,"s3://crabby-images/b7173/b7173ff4cf3b93742da1fa7e3e058aac34b0e225" alt="{a}^' {a}= {{r}}^'_2 {{r}}_2"
after pivoting the rows and columns of
data:image/s3,"s3://crabby-images/db7db/db7db8c1b4c18e2c6f754913728ad5f09fc577f1" alt="{a}^' {a}"
.
Here is the code:
ord = j(n,1,0);
call qr(q,r2,pivqr,lindqr,a,ord);
ss2 = ssq(aa[pivqr,pivqr] - r2` * r2);
print ss2 r2 [format=best6.];
Using the
RUPDT subroutine for stepwise updating of
data:image/s3,"s3://crabby-images/6c306/6c306ac8b358fdb51acfee12aaad5d23d6d0633e" alt="{{r}}"
by the
data:image/s3,"s3://crabby-images/be1ac/be1acfb5e2ef8c1161442c83f8d9b64b570626ff" alt="m"
rows of
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
finally results in an upper triangular
matrix
data:image/s3,"s3://crabby-images/24248/242486abbd8b12cd13a8338f8204d7daaa91b99c" alt="{{r}}_3"
with
data:image/s3,"s3://crabby-images/15aa4/15aa4787908c911810b7e10d8757bac6effbf09c" alt="n-r"
nearly zero diagonal elements.
However, other elements in rows with nearly zero
diagonal elements can have significant values.
The following statements verify that
data:image/s3,"s3://crabby-images/e6103/e61032e5f03746c5574817f2a907042aaeca2e04" alt="{a}^' {a}= {{r}}^'_3 {{r}}_3"
:
r3 = shape(0,n,n);
call rupdt(rup,bup,sup,r3,a);
r3 = rup;
ss3 = ssq(aa - r3` * r3);
print ss3 r3 [format=best6.];
The result
data:image/s3,"s3://crabby-images/24248/242486abbd8b12cd13a8338f8204d7daaa91b99c" alt="{{r}}_3"
of the
RUPDT subroutine can be transformed
into the result
data:image/s3,"s3://crabby-images/b67d3/b67d34d035d19b433c37bce622ddd4c8f9d0de1a" alt="{{r}}_1"
of the
ROOT function by left
applications of Givens rotations to zero out the remaining
significant elements of rows with
small diagonal elements.
Applying the RZLIND subroutine on the upper triangular
result
data:image/s3,"s3://crabby-images/24248/242486abbd8b12cd13a8338f8204d7daaa91b99c" alt="{{r}}_3"
of the
RUPDT subroutine generates a
Cholesky factor
data:image/s3,"s3://crabby-images/2e444/2e444cc7b97399fbd014b0f5ca2cd44e9d3da348" alt="{{r}}_4"
with zero rows corresponding to
diagonal elements that are small, giving the same result
as the
ROOT function (except for the sign of rows) if its
singularity criterion recognizes the same linear dependencies.
Here is the code:
call rzlind(lind,r4,bup,r3);
ss4 = ssq(aa - r4` * r4);
print ss4 r4 [format=best6.];
Consider the rank-deficient linear least squares problem:
data:image/s3,"s3://crabby-images/a6623/a66233e8ed7bbde6919bb7958fdd82a17fc422d8" alt="\min_{{x}} \vert {a}{x}- {b}\vert^2 { where } {a}\in {\cal r}^{m x n}, {\rm rank}({a})=r, r \leq n \leq m"
For
data:image/s3,"s3://crabby-images/81fbc/81fbc09b9a42376dd5fb5ae8420e3d8ec6962699" alt="r=n"
, the optimal solution,
data:image/s3,"s3://crabby-images/97120/97120a74e2bff4dcaa544dec2f357da5158e2a32" alt="\hat{{x}}"
, is unique; however,
for
data:image/s3,"s3://crabby-images/190db/190dbc06916291d2c07b83755bb2d22996d848d3" alt="r \lt n"
, the rank-deficient linear least squares problem has many
optimal solutions, each of which has the same least squares residual
sum of squares:
data:image/s3,"s3://crabby-images/b628a/b628a6a5fc7f986523cff305586a21ce23c75ddc" alt="{ss} = ({a}\hat{{x}} - {b})^'({a}\hat{{x}} - {b})"
The solution of the full-rank problem,
data:image/s3,"s3://crabby-images/81fbc/81fbc09b9a42376dd5fb5ae8420e3d8ec6962699" alt="r=n"
,
is illustrated in the
QR call.
The following list shows several
solutions to the singular problem.
The following example uses the
data:image/s3,"s3://crabby-images/d6b2f/d6b2f2e439cedd2fe3f3119fedc56c7b2571e6d7" alt="12 x 8"
matrix from the
preceding section and generates a new column vector
data:image/s3,"s3://crabby-images/e95c3/e95c3adeb536861135abe117acbe60c24b47c239" alt="{b}"
.
The vector
data:image/s3,"s3://crabby-images/e95c3/e95c3adeb536861135abe117acbe60c24b47c239" alt="{b}"
and the matrix
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
are shown in the output.
b = uniform(j(12,1,1));
ab = a` * b;
print b a [format=best6.];
Each entry in the following list solves the
rank-deficient linear least squares problem.
Note that while each method minimizes the residual sum of squares,
not all of the given solutions are of minimum Euclidean length.
- Use the singular value decomposition of
, given by
.
Take the reciprocals of significant singular
values and set the small values of
to zero.
call svd(u,d,v,a);
t = 1e-12 * d[1];
do i=1 to n;
if d[i] < t then d[i] = 0.;
else d[i] = 1. / d[i];
end;
x1 = v * diag(d) * u` * b;
len1 = x1` * x1;
ss1 = ssq(a * x1 - b);
x1 = x1`;
print ss1 len1, x1 [format=best6.];
The solution
obtained by singular value
decomposition,
, is of minimum Euclidean length.
- Use QR decomposition with column pivoting:
![{a}{{{{\pi}}}}= {q}{{r}}= [ {y}& {z} ] [ {{r}}_1 & {{r}}_2 \ 0 & 0 ] = {y}[ {{r}}_1 & {{r}}_2 ]](images/langref_langrefeq1130.gif)
Set the right part
to zero and invert the upper
triangular matrix
to obtain a generalized
inverse
and an optimal solution
:
![{{r}}^- = [ {{r}}_1^{-1} \ 0 ] \hspace*{0.25in} \hat{{x}}_2 = {{{{\pi}}}}{{r}}^- {y}^' {b}](images/langref_langrefeq1134.gif)
ord = j(n,1,0);
call qr(qtb,r2,pivqr,lindqr,a,ord,b);
nr = n - lindqr;
r = r2[1:nr,1:nr];
x2 = shape(0,n,1);
x2[pivqr] = trisolv(1,r,qtb[1:nr]) // j(lindqr,1,0.);
len2 = x2` * x2;
ss2 = ssq(a * x2 - b);
x2 = x2`;
print ss2 len2, x2 [format=best6.];
Note that the residual sum of squares is minimal, but the
solution
is not of minimum Euclidean length.
- Use the result
of the ROOT function on
this page to obtain the vector piv indicating
the zero rows in the upper triangular matrix
:
r1 = root(aa);
nr = n - lind;
piv = shape(0,n,1);
j1 = 1; j2 = nr + 1;
do i=1 to n;
if r1[i,i] ^= 0 then do;
piv[j1] = i; j1 = j1 + 1;
end;
else do;
piv[j2] = i; j2 = j2 + 1;
end;
end;
Now compute
by solving the equation
.
r = r1[piv[1:nr],piv[1:nr]];
x = trisolv(2,r,ab[piv[1:nr]]);
x = trisolv(1,r,x);
x3 = shape(0,n,1);
x3[piv] = x // j(lind,1,0.);
len3 = x3` * x3;
ss3 = ssq(a * x3 - b);
x3 = x3`;
print ss3 len3, x3 [format=best6.];
Note that the residual sum of squares is minimal, but the
solution
is not of minimum Euclidean length.
- Use the result
of the RUPDT call on
this page and the vector piv (obtained
in the previous solution), which indicates the zero
rows of upper triangular matrices
and
.
After zeroing out the rows of
belonging
to small diagonal pivots, solve the system
.
r3 = shape(0,n,n);
qtb = shape(0,n,1);
call rupdt(rup,bup,sup,r3,a,qtb,b);
r3 = rup; qtb = bup;
call rzlind(lind,r4,bup,r3,,qtb);
qtb = bup[piv[1:nr]];
x = trisolv(1,r4[piv[1:nr],piv[1:nr]],qtb);
x4 = shape(0,n,1);
x4[piv] = x // j(lind,1,0.);
len4 = x4` * x4;
ss4 = ssq(a * x4 - b);
x4 = x4`;
print ss4 len4, x4 [format=best6.];
Since the matrices
and
are the
same (except for the signs of rows), the solution
is the same as
. - Use the result
of the RZLIND call in the
previous solution, which is the result of the first
step of complete QR decomposition, and perform
the second step of complete QR decomposition.
The rows of matrix
can be
permuted to the upper trapezoidal form
![[ \hat{{{r}}} & {t}\ 0 & 0 ]](images/langref_langrefeq1139.gif)
where
is nonsingular and
upper triangular and
is rectangular.
Next, perform the second step of complete QR
decomposition with the lower triangular matrix
![[ \hat{{{r}}}^' \ {t}^' ] = \bar{{y}} [ \bar{{{r}}} \ 0 ]](images/langref_langrefeq1142.gif)
which leads to the upper triangular matrix
.
r = r4[piv[1:nr],]`;
call qr(q,r5,piv2,lin2,r);
y = trisolv(2,r5,qtb);
x5 = q * (y // j(lind,1,0.));
len5 = x5` * x5;
ss5 = ssq(a * x5 - b);
x5 = x5`;
print ss5 len5, x5 [format=best6.];
The solution
obtained by complete
QR decomposition has minimum Euclidean length.
- Perform both steps of complete QR decomposition.
The first step performs the pivoted QR decomposition of
,
![{a}{{{{\pi}}}}= {q}{{r}}= {y}[ {{r}}\ 0 ] = {y}[ \hat{{{r}}}{t}\ 0 ]](images/langref_langrefeq1145.gif)
where
is nonsingular and
upper triangular and
is rectangular.
The second step performs a QR decomposition
as described in the previous method.
This results in
![{a}{{{{\pi}}}}= {y}[ \bar{{{r}}}^' & 0 \ 0 & 0 ] \bar{{y}}^'](images/langref_langrefeq1146.gif)
where
is lower triangular.
ord = j(n,1,0);
call qr(qtb,r2,pivqr,lindqr,a,ord,b);
nr = n - lindqr;
r = r2[1:nr,]`;
call qr(q,r5,piv2,lin2,r);
y = trisolv(2,r5,qtb[1:nr]);
x6 = shape(0,n,1);
x6[pivqr] = q * (y // j(lindqr,1,0.));
len6 = x6` * x6;
ss6 = ssq(a * x6 - b);
x6 = x6`;
print ss6 len6, x6 [format=best6.];
The solution
obtained by complete
QR decomposition has minimum Euclidean length.
- Perform complete QR decomposition
with the QR and LUPDT calls:
ord = j(n,1,0);
call qr(qtb,r2,pivqr,lindqr,a,ord,b);
nr = n - lindqr;
r = r2[1:nr,1:nr]`; z = r2[1:nr,nr+1:n]`;
call lupdt(lup,bup,sup,r,z);
rd = trisolv(3,lup,r2[1:nr,]);
rd = trisolv(4,lup,rd);
x7 = shape(0,n,1);
x7[pivqr] = rd` * qtb[1:nr,];
len7 = x7` * x7;
ss7 = ssq(a * x7 - b);
x7 = x7`;
print ss7 len7, x7 [format=best6.];
The solution
obtained by complete
QR decomposition has minimum Euclidean length.
- Perform complete QR decomposition with
the RUPDT, RZLIND, and LUPDT calls:
r3 = shape(0,n,n);
qtb = shape(0,n,1);
call rupdt(rup,bup,sup,r3,a,qtb,b);
r3 = rup; qtb = bup;
call rzlind(lind,r4,bup,r3,,qtb);
nr = n - lind; qtb = bup;
r = r4[piv[1:nr],piv[1:nr]]`;
z = r4[piv[1:nr],piv[nr+1:n]]`;
call lupdt(lup,bup,sup,r,z);
rd = trisolv(3,lup,r4[piv[1:nr],]);
rd = trisolv(4,lup,rd);
x8 = shape(0,n,1);
x8 = rd` * qtb[piv[1:nr],];
len8 = x8` * x8;
ss8 = ssq(a * x8 - b);
x8 = x8`;
print ss8 len8, x8 [format=best6.];
The solution
obtained by complete
QR decomposition has minimum Euclidean length.
The same result can be obtained
with the APPCORT or COMPORT call.
You can use various methods to compute the Moore-Penrose inverse
data:image/s3,"s3://crabby-images/159d4/159d489a7b9bd06cd23869a1b9bd3057496dd437" alt="{a}^-"
of a rectangular matrix
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
using orthogonal methods.
The entries in the following list find the Moore-Penrose
inverse of the matrix
data:image/s3,"s3://crabby-images/c3fe0/c3fe0b4fa985b96bb47c099e4c3bdd223be6095b" alt="{a}"
shown on
this page.
- Use the GINV operator.
The GINV operator in IML uses the singular
decomposition
.
The result
should be
identical to the result given by the next solution.
ga = ginv(a);
t1 = a * ga; t2 = t1`;
t3 = ga * a; t4 = t3`;
ss1 = ssq(t1 - t2) + ssq(t3 - t4) +
ssq(t1 * a - a) + ssq(t3 * ga - ga);
print ss1, ga [format=best6.];
- Use singular value decomposition.
The singular decomposition
with
,
,
and
, can be
used to compute
,
with
and
data:image/s3,"s3://crabby-images/85aec/85aecc3a845664c6be7ad2b0d978ff47155e2b15" alt="d_i^\dagger = \{ 0 & {where } d_i \leq \epsilon \ 1/d_i & {otherwise} ."
The result
should be the same as that
given by the GINV operator if the singularity
criterion
is selected correspondingly.
Since you cannot specify the criterion
for the GINV operator, the singular value decomposition
approach can be important for applications where the
GINV operator uses an unsuitable
criterion.
The slight discrepancy between the values of
SS1 and SS2 is due to rounding that occurs
in the statement that computes the matrix GA.
call svd(u,d,v,a);
do i=1 to n;
if d[i] <= 1e-10 * d[1] then d[i] = 0.;
else d[i] = 1. / d[i];
end;
ga = v * diag(d) * u`;
t1 = a * ga; t2 = t1`;
t3 = ga * a; t4 = t3`;
ss2 = ssq(t1 - t2) + ssq(t3 - t4) +
ssq(t1 * a - a) + ssq(t3 * ga - ga);
print ss2;
- Use complete QR decomposition.
The complete QR decomposition
![{a}= {y}[ \bar{{{r}}}^' & 0 \ 0 & 0 ] \bar{{y}}^' {{{{\pi}}}}^'](images/langref_langrefeq1158.gif)
where
is lower
triangular, yields the Moore-Penrose inverse
![{a}^- = {{{{\pi}}}}\bar{{y}} [ \bar{{{r}}}^{-' & 0 \ 0 & 0 ] {y}^'](images/langref_langrefeq1159.gif)
ord = j(n,1,0);
call qr(q1,r2,pivqr,lindqr,a,ord);
nr = n - lindqr;
q1 = q1[,1:nr]; r = r2[1:nr,]`;
call qr(q2,r5,piv2,lin2,r);
tt = trisolv(4,r5`,q1`);
ga = shape(0,n,m);
ga[pivqr,] = q2 * (tt // shape(0,n-nr,m));
t1 = a * ga; t2 = t1`;
t3 = ga * a; t4 = t3`;
ss3 = ssq(t1 - t2) + ssq(t3 - t4) +
ssq(t1 * a - a) + ssq(t3 * ga - ga);
print ss3;
- Use complete QR decomposition with QR and LUPDT:
ord = j(n,1,0);
call qr(q,r2,pivqr,lindqr,a,ord);
nr = n - lindqr;
r = r2[1:nr,1:nr]`; z = r2[1:nr,nr+1:n]`;
call lupdt(lup,bup,sup,r,z);
rd = trisolv(3,lup,r2[1:nr,]);
rd = trisolv(4,lup,rd);
ga = shape(0,n,m);
ga[pivqr,] = rd` * q[,1:nr]`;
t1 = a * ga; t2 = t1`;
t3 = ga * a; t4 = t3`;
ss4 = ssq(t1 - t2) + ssq(t3 - t4) +
ssq(t1 * a - a) + ssq(t3 * ga - ga);
print ss4;
Use complete QR decomposition with RUPDT and LUPDT:
r3 = shape(0,n,n);
y = i(m); qtb = shape(0,n,m);
call rupdt(rup,bup,sup,r3,a,qtb,y);
r3 = rup; qtb = bup;
call rzlind(lind,r4,bup,r3,,qtb);
nr = n - lind; qtb = bup;
r = r4[piv[1:nr],piv[1:nr]]`;
z = r4[piv[1:nr],piv[nr+1:n]]`;
call lupdt(lup,bup,sup,r,z);
rd = trisolv(3,lup,r4[piv[1:nr],]);
rd = trisolv(4,lup,rd);
ga = shape(0,n,m);
ga = rd` * qtb[piv[1:nr],];
t1 = a * ga; t2 = t1`;
t3 = ga * a; t4 = t3`;
ss5 = ssq(t1 - t2) + ssq(t3 - t4) +
ssq(t1 * a - a) + ssq(t3 * ga - ga);
print ss5;