LSQR finds a solution x to the following problems:
where A is a matrix with m rows and n columns, b is an
m-vector, and damp
is a scalar. (All quantities are real.)
The matrix A is intended to be large and sparse. It is accessed
by means of subroutine calls to aprod
.
The rhs vector b is input via u, and subsequently overwritten.
Note: LSQR uses an iterative method to approximate the solution. The number of iterations required to reach a certain accuracy depends strongly on the scaling of the problem. Poor scaling of the rows or columns of A should therefore be avoided where possible.
For example, in problem 1 the solution is unaltered by row-scaling. If a row of A is very small or large compared to the other rows of A, the corresponding row of ( A b ) should be scaled up or down.
In problems 1 and 2, the solution x is easily recovered following column-scaling. Unless better information is known, the nonzero columns of A should be scaled so that they all have the same Euclidean norm (e.g., 1.0).
In problem 3, there is no freedom to re-scale if damp is nonzero. However, the value of damp should be assigned only after attention has been paid to the scaling of A.
The parameter damp is intended to help regularize ill-conditioned systems, by preventing the true solution from being very large. Another aid to regularization is provided by the parameter acond, which may be used to terminate iterations before the computed solution becomes very large.
Note that x is not an input parameter. If some initial estimate x0 is known and if damp = 0, one could proceed as follows:
This requires that x0 be available before and after the call to LSQR. To judge the benefits, suppose LSQR takes k1 iterations to solve Ax = b and k2 iterations to solve Adx = r0. If x0 is "good", norm(r0) will be smaller than norm(b). If the same stopping tolerances atol and btol are used for each system, k1 and k2 will be similar, but the final solution x0 + dx should be more accurate. The only way to reduce the total work is to use a larger stopping tolerance for the second system. If some value btol is suitable for Ax = b, the larger value btolnorm(b)/norm(r0) should be suitable for A*dx = r0.
Preconditioning is another way to reduce the number of iterations. If it is possible to solve a related system Mx = b efficiently, where M approximates A in some helpful way (e.g. M - A has low rank or its elements are small relative to those of A), LSQR may converge more rapidly on the system AM(inverse)z = b, after which x can be recovered by solving Mx = z.
NOTE: If A is symmetric, LSQR should not be used! Alternatives are the symmetric conjugate-gradient method (cg) and/or SYMMLQ. SYMMLQ is an implementation of symmetric cg that applies to any symmetric A and will converge more rapidly than LSQR. If A is positive definite, there are other implementations of symmetric cg that require slightly less work per iteration than SYMMLQ (but will take the same number of iterations).
The following quantities are used in discussing the subroutine parameters:
Abar = ( A ), bbar = ( b )
( damp*I ) ( 0 )
r = b - A*x, rbar = bbar - Abar*x
rnorm = sqrt( norm(r)**2 + damp**2 * norm(x)**2 )
= norm( rbar )
relpr = the relative precision of floating-point arithmetic
on the machine being used. On most machines,
relpr is about 1.0e-7 and 1.0d-16 in single and double
precision respectively.
LSQR minimizes the function rnorm
with respect to x
.
C.C. Paige and M.A. Saunders, LSQR: An algorithm for sparse linear equations and sparse least squares, ACM Transactions on Mathematical Software 8, 1 (March 1982), pp. 43-71.
C.C. Paige and M.A. Saunders, Algorithm 583, LSQR: Sparse linear equations and least-squares problems, ACM Transactions on Mathematical Software 8, 2 (June 1982), pp. 195-209.
C.L. Lawson, R.J. Hanson, D.R. Kincaid and F.T. Krogh, Basic linear algebra subprograms for Fortran usage, ACM Transactions on Mathematical Software 5, 3 (Sept 1979), pp. 308-323 and 324-325.
Note
The number of iterations required by LSQR will usually decrease if the computation is performed in higher precision.
Type | Intent | Optional | Attributes | Name | ||
---|---|---|---|---|---|---|
class(lsqr_solver), | intent(inout) | :: | me | |||
integer, | intent(in) | :: | m |
the number of rows in |
||
integer, | intent(in) | :: | n |
the number of columns in |
||
real(kind=wp), | intent(in) | :: | damp |
The damping parameter for problem 3 above.
(damp should be 0.0 for problems 1 and 2.)
If the system The work per iteration and the storage needed
by LSQR are the same for all values of |
||
logical, | intent(in) | :: | wantse |
A logical variable to say if the array |
||
real(kind=wp), | intent(inout) | :: | u(m) |
The rhs vector |
||
real(kind=wp), | intent(inout) | :: | v(n) |
workspace |
||
real(kind=wp), | intent(inout) | :: | w(n) |
workspace |
||
real(kind=wp), | intent(out) | :: | x(n) |
Returns the computed solution |
||
real(kind=wp), | intent(out), | dimension(*) | :: | se |
If If |
|
real(kind=wp), | intent(in) | :: | atol |
An estimate of the relative error in the data
defining the matrix |
||
real(kind=wp), | intent(in) | :: | btol |
An estimate of the relative error in the data
defining the rhs vector |
||
real(kind=wp), | intent(in) | :: | conlim |
An upper limit on
Normally, Suggested value:
Note: If the user is not concerned about the parameters
|
||
integer, | intent(in) | :: | itnlim |
An upper limit on the number of iterations.
Suggested value:
* |
||
integer, | intent(in) | :: | nout |
File number for printed output. If nonzero,
a summary will be printed on file |
||
integer, | intent(out) | :: | istop |
An integer giving the reason for termination:
|
||
integer, | intent(out) | :: | itn |
The number of iterations performed. |
||
real(kind=wp), | intent(out) | :: | anorm |
An estimate of the Frobenius norm of |
||
real(kind=wp), | intent(out) | :: | acond |
An estimate of |
||
real(kind=wp), | intent(out) | :: | rnorm |
An estimate of the final value of |
||
real(kind=wp), | intent(out) | :: | arnorm |
An estimate of the final value of
|
||
real(kind=wp), | intent(out) | :: | xnorm |
An estimate of the norm of the final
solution vector |
subroutine LSQR ( me, m, n, damp, wantse, & u, v, w, x, se, & atol, btol, conlim, itnlim, nout, & istop, itn, anorm, acond, rnorm, arnorm, xnorm) class(lsqr_solver),intent(inout) :: me integer,intent(in) :: m !! the number of rows in `A`. integer,intent(in) :: n !! the number of columns in `A`. real(wp),intent(in) :: damp !! The damping parameter for problem 3 above. !! (damp should be 0.0 for problems 1 and 2.) !! If the system `A*x = b` is incompatible, values !! of `damp` in the range 0 to `sqrt(relpr)*norm(A)` !! will probably have a negligible effect. !! Larger values of `damp` will tend to decrease !! the norm of `x` and reduce the number of !! iterations required by LSQR. !! !! The work per iteration and the storage needed !! by LSQR are the same for all values of `damp`. logical,intent(in) :: wantse !! A logical variable to say if the array `se(*)` !! of standard error estimates should be computed. !! If `m > n` or `damp > 0`, the system is !! overdetermined and the standard errors may be !! useful. (See the first LSQR reference.) !! Otherwise (`m <= n` and `damp = 0`) they do not !! mean much. Some time and storage can be saved !! by setting `wantse = .false.` and using any !! convenient array for `se(*)`, which won't be !! touched. real(wp),intent(inout):: u(m) !! The rhs vector `b`. Beware that `u` is !! over-written by LSQR. real(wp),intent(inout):: v(n) !! workspace real(wp),intent(inout):: w(n) !! workspace real(wp),intent(out) :: x(n) !! Returns the computed solution `x`. real(wp),dimension(*),intent(out) :: se !! If `wantse` is true, the dimension of `se` must be !! `n` or more. `se(*)` then returns standard error !! estimates for the components of `x`. !! For each `i`, `se(i)` is set to the value !! `rnorm * sqrt( sigma(i,i) / t )`, !! where `sigma(i,i)` is an estimate of the i-th !! diagonal of the inverse of `Abar(transpose)*Abar` !! and: !! * `t = 1 if m <= n` !! * `t = m - n if m > n and damp = 0` !! * `t = m if damp /= 0` !! !! If `wantse` is false, `se(*)` will not be touched. !! The actual parameter can be any suitable array !! of any length. real(wp),intent(in) :: atol !! An estimate of the relative error in the data !! defining the matrix `A`. For example, !! if `A` is accurate to about 6 digits, set !! `atol = 1.0e-6`. real(wp),intent(in) :: btol !! An estimate of the relative error in the data !! defining the rhs vector `b`. For example, !! if `b` is accurate to about 6 digits, set !! `btol = 1.0e-6`. real(wp),intent(in) :: conlim !! An upper limit on `cond(Abar)`, the apparent !! condition number of the matrix `Abar`. !! Iterations will be terminated if a computed !! estimate of `cond(Abar)` exceeds `conlim`. !! This is intended to prevent certain small or !! zero singular values of `A` or `Abar` from !! coming into effect and causing unwanted growth !! in the computed solution. !! !! `conlim` and `damp` may be used separately or !! together to regularize ill-conditioned systems. !! !! Normally, `conlim` should be in the range !! 1000 to `1/relpr`. !! !! Suggested value: !! !! * `conlim = 1/(100*relpr)` for compatible systems, !! * `conlim = 1/(10*sqrt(relpr))` for least squares. !! !! Note: If the user is not concerned about the parameters !! `atol`, `btol` and `conlim`, any or all of them may be set !! to zero. The effect will be the same as the values !! `relpr`, `relpr` and `1/relpr` respectively. integer,intent(in) :: itnlim !! An upper limit on the number of iterations. !! Suggested value: !! * `itnlim = n/2` for well-conditioned systems !! with clustered singular values, !! * `itnlim = 4*n` otherwise. integer,intent(in) :: nout !! File number for printed output. If nonzero, !! a summary will be printed on file `nout`. integer,intent(out) :: istop !! An integer giving the reason for termination: !! !! * 0 -- `x` = 0 is the exact solution. !! No iterations were performed. !! * 1 -- The equations `A*x = b` are probably !! compatible. `Norm(A*x - b)` is sufficiently !! small, given the values of `atol` and `btol`. !! * 2 -- `damp` is zero. The system `A*x = b` is probably !! not compatible. A least-squares solution has !! been obtained that is sufficiently accurate, !! given the value of `atol`. !! * 3 -- `damp` is nonzero. A damped least-squares !! solution has been obtained that is sufficiently !! accurate, given the value of `atol`. !! * 4 -- An estimate of `cond(Abar)` has exceeded !! `conlim`. The system `A*x = b` appears to be !! ill-conditioned. Otherwise, there could be an !! error in subroutine `aprod`. !! * 5 -- The iteration limit `itnlim` was reached. integer,intent(out) :: itn !! The number of iterations performed. real(wp),intent(out) :: anorm !! An estimate of the Frobenius norm of `Abar`. !! This is the square-root of the sum of squares !! of the elements of `Abar`. !! If `damp` is small and if the columns of `A` !! have all been scaled to have length 1.0, !! `anorm` should increase to roughly `sqrt(n)`. !! A radically different value for `anorm` may !! indicate an error in subroutine `aprod` (there !! may be an inconsistency between modes 1 and 2). real(wp),intent(out) :: acond !! An estimate of `cond(Abar)`, the condition !! number of `Abar`. A very high value of `acond` !! may again indicate an error in `aprod`. real(wp),intent(out) :: rnorm !! An estimate of the final value of `norm(rbar)`, !! the function being minimized (see notation !! above). This will be small if `A*x = b` has !! a solution. real(wp),intent(out) :: arnorm !! An estimate of the final value of !! `norm( Abar(transpose)*rbar )`, the norm of !! the residual for the usual normal equations. !! This should be small in all cases. (arnorm !! will often be smaller than the true value !! computed from the output vector `x`.) real(wp),intent(out) :: xnorm !! An estimate of the norm of the final !! solution vector `x`. logical :: damped integer :: i, maxdx, nconv, nstop real(wp) :: alfopt, alpha, beta, bnorm, & cs, cs1, cs2, ctol, & delta, dknorm, dnorm, dxk, dxmax, & gamma, gambar, phi, phibar, psi, & res2, rho, rhobar, rhbar1, & rhs, rtol, sn, sn1, sn2, & t, tau, temp, test1, test2, test3, & theta, t1, t2, t3, xnorm1, z, zbar logical :: print_iter logical,parameter :: extra = .true. !! for extra printing below. character(len=*),parameter :: enter = ' Enter LSQR. ' character(len=*),parameter :: exit = ' Exit LSQR. ' character(len=*),dimension(0:5),parameter :: msg = [ 'The exact solution is x = 0 ',& 'A solution to Ax = b was found, given atol, btol ',& 'A least-squares solution was found, given atol ',& 'A damped least-squares solution was found, given atol',& 'Cond(Abar) seems to be too large, given conlim ',& 'The iteration limit was reached ' ] ! Initialize. if (nout /= 0) then write(nout,'(//A)') enter//' Least-squares solution of Ax = b' write(nout,'(A,I7,A,I7,A)') ' The matrix A has', m, ' rows and', n, ' columns' write(nout,'(1P,A,E22.14,3X,A,L10)') ' damp =', damp, 'wantse =', wantse write(nout,'(1P,A,E10.2,15x,A,E10.2)') ' atol =', atol, 'conlim =', conlim write(nout,'(1P,A,E10.2,15x,A,I10)') ' btol =', btol, 'itnlim =', itnlim end if damped = damp > zero itn = 0 istop = 0 nstop = 0 maxdx = 0 if (conlim > zero) then ctol = one / conlim else ctol = zero end if anorm = zero acond = zero dnorm = zero dxmax = zero res2 = zero psi = zero xnorm = zero xnorm1 = zero cs2 = - one sn2 = zero z = zero ! Set up the first vectors u and v for the bidiagonalization. ! These satisfy beta*u = b, alpha*v = A(transpose)*u. do i = 1, n v(i) = zero x(i) = zero end do if ( wantse ) then do i = 1, n se(i) = zero end do end if alpha = zero beta = dnrm2 ( m, u, 1 ) if (beta > zero) then call dscal ( m, (one / beta), u, 1 ) call me%aprod ( 2, m, n, v, u ) alpha = dnrm2 ( n, v, 1 ) end if if (alpha > zero) then call dscal ( n, (one / alpha), v, 1 ) call dcopy ( n, v, 1, w, 1 ) end if arnorm = alpha * beta if (arnorm /= zero) then rhobar = alpha phibar = beta bnorm = beta rnorm = beta if (nout /= 0) then if ( damped ) then write(nout, '(//A)') & ' Itn x(1) Function Compatible LS Norm Abar Cond Abar' else write(nout, '(//A)') & ' Itn x(1) Function Compatible LS Norm A Cond A' end if test1 = one test2 = alpha / beta if ( extra ) then write(nout, '(80X,A)') ' phi dknorm dxk alfa_opt' end if write(nout, '(1P, I6, 2E17.9, 4E10.2, E9.1, 3E8.1)') itn, x(1), rnorm, test1, test2 write(nout, '(A)') '' end if do ! Main iteration loop. itn = itn + 1 ! Perform the next step of the bidiagonalization to obtain the ! next beta, u, alpha, v. These satisfy the relations ! beta*u = A*v - alpha*u, ! alpha*v = A(transpose)*u - beta*v. call dscal ( m, (- alpha), u, 1 ) call me%aprod ( 1, m, n, v, u ) beta = dnrm2 ( m, u, 1 ) ! Accumulate anorm = || Bk || = sqrt( sum of alpha**2 + beta**2 + damp**2 ). temp = d2norm( alpha, beta ) temp = d2norm( temp , damp ) anorm = d2norm( anorm, temp ) if (beta > zero) then call dscal ( m, (one / beta), u, 1 ) call dscal ( n, (- beta), v, 1 ) call me%aprod ( 2, m, n, v, u ) alpha = dnrm2 ( n, v, 1 ) if (alpha > zero) then call dscal ( n, (one / alpha), v, 1 ) end if end if ! Use a plane rotation to eliminate the damping parameter. ! This alters the diagonal (rhobar) of the lower-bidiagonal matrix. rhbar1 = rhobar if ( damped ) then rhbar1 = d2norm( rhobar, damp ) cs1 = rhobar / rhbar1 sn1 = damp / rhbar1 psi = sn1 * phibar phibar = cs1 * phibar end if ! Use a plane rotation to eliminate the subdiagonal element (beta) ! of the lower-bidiagonal matrix, giving an upper-bidiagonal matrix. rho = d2norm( rhbar1, beta ) cs = rhbar1 / rho sn = beta / rho theta = sn * alpha rhobar = - cs * alpha phi = cs * phibar phibar = sn * phibar tau = sn * phi ! Update x, w and (perhaps) the standard error estimates. t1 = phi / rho t2 = - theta / rho t3 = one / rho dknorm = zero if ( wantse ) then do i = 1, n t = w(i) x(i) = t1*t + x(i) w(i) = t2*t + v(i) t = (t3*t)**2 se(i) = t + se(i) dknorm = t + dknorm end do else do i = 1, n t = w(i) x(i) = t1*t + x(i) w(i) = t2*t + v(i) dknorm = (t3*t)**2 + dknorm end do end if ! Monitor the norm of d_k, the update to x. ! dknorm = norm( d_k ) ! dnorm = norm( D_k ), where D_k = (d_1, d_2, ..., d_k ) ! dxk = norm( phi_k d_k ), where new x = x_k + phi_k d_k. dknorm = sqrt( dknorm ) dnorm = d2norm( dnorm, dknorm ) dxk = abs( phi * dknorm ) if (dxmax < dxk ) then dxmax = dxk maxdx = itn end if ! Use a plane rotation on the right to eliminate the ! super-diagonal element (theta) of the upper-bidiagonal matrix. ! Then use the result to estimate norm(x). delta = sn2 * rho gambar = - cs2 * rho rhs = phi - delta * z zbar = rhs / gambar xnorm = d2norm( xnorm1, zbar ) gamma = d2norm( gambar, theta ) cs2 = gambar / gamma sn2 = theta / gamma z = rhs / gamma xnorm1 = d2norm( xnorm1, z ) ! Test for convergence. ! First, estimate the norm and condition of the matrix Abar, ! and the norms of rbar and Abar(transpose)*rbar. acond = anorm * dnorm res2 = d2norm( res2 , psi ) rnorm = d2norm( res2 , phibar ) arnorm = alpha * abs( tau ) ! Now use these norms to estimate certain other quantities, ! some of which will be small near a solution. alfopt = sqrt( rnorm / (dnorm * xnorm) ) test1 = rnorm / bnorm test2 = zero if (rnorm > zero) test2 = arnorm / (anorm * rnorm) test3 = one / acond t1 = test1 / (one + anorm * xnorm / bnorm) rtol = btol + atol * anorm * xnorm / bnorm ! The following tests guard against extremely small values of ! atol, btol or ctol. (The user may have set any or all of ! the parameters atol, btol, conlim to zero.) ! The effect is equivalent to the normal tests using ! atol = relpr, btol = relpr, conlim = 1/relpr. t3 = one + test3 t2 = one + test2 t1 = one + t1 if (itn >= itnlim) istop = 5 if (t3 <= one ) istop = 4 if (t2 <= one ) istop = 2 if (t1 <= one ) istop = 1 ! Allow for tolerances set by the user. if (test3 <= ctol) istop = 4 if (test2 <= atol) istop = 2 if (test1 <= rtol) istop = 1 ! See if it is time to print something. if (nout /= 0) then print_iter = (n <= 40 ) .or. & (itn <= 10 ) .or. & (itn >= itnlim-10) .or. & (mod(itn,10) == 0 ) .or. & (test3 <= 2.0*ctol) .or. & (test2 <= 10.0*atol) .or. & (test1 <= 10.0*rtol) .or. & (istop /= 0 ) if (print_iter) then ! Print a line for this iteration. ! "extra" is for experimental purposes. if ( extra ) then write(nout, '(1P, I6, 2E17.9, 4E10.2, E9.1, 3E8.1)') & itn, x(1), rnorm, test1, test2, anorm, acond, phi, dknorm, dxk, alfopt else write(nout, '(1P, I6, 2E17.9, 4E10.2, E9.1, 3E8.1)') & itn, x(1), rnorm, test1, test2, anorm, acond end if !if (mod(itn,10) == 0) write(nout, '(A)') '' end if end if ! Stop if appropriate. ! The convergence criteria are required to be met on nconv ! consecutive iterations, where nconv is set below. ! Suggested value: nconv = 1, 2 or 3. if (istop == 0) then nstop = 0 else nconv = 1 nstop = nstop + 1 if (nstop < nconv .and. itn < itnlim) istop = 0 end if if (istop /= 0) exit end do ! End of iteration loop. ! Finish off the standard error estimates. if ( wantse ) then t = one if (m > n) t = m - n if ( damped ) t = m t = rnorm / sqrt( t ) do i = 1, n se(i) = t * sqrt( se(i) ) end do end if end if ! Decide if istop = 2 or 3. ! Print the stopping condition. if (damped .and. istop == 2) istop = 3 if (nout /= 0) then write(nout, '(//A,5X,A,I2,15X,A,I8)') exit, 'istop =', istop, 'itn =', itn write(nout, '(1P,A,5X,A,E12.5,5X,A,E12.5)') exit, 'anorm =', anorm, 'acond =', acond write(nout, '(1P,A,5X,A,E12.5,5X,A,E12.5)') exit, 'bnorm =', bnorm, 'xnorm =', xnorm write(nout, '(1P,A,5X,A,E12.5,5X,A,E12.5)') exit, 'rnorm =', rnorm, 'arnorm =', arnorm write(nout, '(1P,A,5X,A,E8.1,A,I8)') exit, 'max dx =', dxmax, ' occurred at itn ',maxdx write(nout, '(1P,A,5X,A,E8.1,A)' ) exit, ' =', dxmax/(xnorm + 1.0e-20_wp), '*xnorm' write(nout, '(A,5X,A)' ) exit, msg(istop) end if end subroutine LSQR