zunt01.f
来自「famous linear algebra library (LAPACK) p」· F 代码 · 共 181 行
F
181 行
SUBROUTINE ZUNT01( ROWCOL, M, N, U, LDU, WORK, LWORK, RWORK,
$ RESID )
*
* -- LAPACK test routine (version 3.1) --
* Univ. of Tennessee, Univ. of California Berkeley and NAG Ltd..
* November 2006
*
* .. Scalar Arguments ..
CHARACTER ROWCOL
INTEGER LDU, LWORK, M, N
DOUBLE PRECISION RESID
* ..
* .. Array Arguments ..
DOUBLE PRECISION RWORK( * )
COMPLEX*16 U( LDU, * ), WORK( * )
* ..
*
* Purpose
* =======
*
* ZUNT01 checks that the matrix U is unitary by computing the ratio
*
* RESID = norm( I - U*U' ) / ( n * EPS ), if ROWCOL = 'R',
* or
* RESID = norm( I - U'*U ) / ( m * EPS ), if ROWCOL = 'C'.
*
* Alternatively, if there isn't sufficient workspace to form
* I - U*U' or I - U'*U, the ratio is computed as
*
* RESID = abs( I - U*U' ) / ( n * EPS ), if ROWCOL = 'R',
* or
* RESID = abs( I - U'*U ) / ( m * EPS ), if ROWCOL = 'C'.
*
* where EPS is the machine precision. ROWCOL is used only if m = n;
* if m > n, ROWCOL is assumed to be 'C', and if m < n, ROWCOL is
* assumed to be 'R'.
*
* Arguments
* =========
*
* ROWCOL (input) CHARACTER
* Specifies whether the rows or columns of U should be checked
* for orthogonality. Used only if M = N.
* = 'R': Check for orthogonal rows of U
* = 'C': Check for orthogonal columns of U
*
* M (input) INTEGER
* The number of rows of the matrix U.
*
* N (input) INTEGER
* The number of columns of the matrix U.
*
* U (input) COMPLEX*16 array, dimension (LDU,N)
* The unitary matrix U. U is checked for orthogonal columns
* if m > n or if m = n and ROWCOL = 'C'. U is checked for
* orthogonal rows if m < n or if m = n and ROWCOL = 'R'.
*
* LDU (input) INTEGER
* The leading dimension of the array U. LDU >= max(1,M).
*
* WORK (workspace) COMPLEX*16 array, dimension (LWORK)
*
* LWORK (input) INTEGER
* The length of the array WORK. For best performance, LWORK
* should be at least N*N if ROWCOL = 'C' or M*M if
* ROWCOL = 'R', but the test will be done even if LWORK is 0.
*
* RWORK (workspace) DOUBLE PRECISION array, dimension (min(M,N))
* Used only if LWORK is large enough to use the Level 3 BLAS
* code.
*
* RESID (output) DOUBLE PRECISION
* RESID = norm( I - U * U' ) / ( n * EPS ), if ROWCOL = 'R', or
* RESID = norm( I - U' * U ) / ( m * EPS ), if ROWCOL = 'C'.
*
* =====================================================================
*
* .. Parameters ..
DOUBLE PRECISION ZERO, ONE
PARAMETER ( ZERO = 0.0D+0, ONE = 1.0D+0 )
* ..
* .. Local Scalars ..
CHARACTER TRANSU
INTEGER I, J, K, LDWORK, MNMIN
DOUBLE PRECISION EPS
COMPLEX*16 TMP, ZDUM
* ..
* .. External Functions ..
LOGICAL LSAME
DOUBLE PRECISION DLAMCH, ZLANSY
COMPLEX*16 ZDOTC
EXTERNAL LSAME, DLAMCH, ZLANSY, ZDOTC
* ..
* .. External Subroutines ..
EXTERNAL ZHERK, ZLASET
* ..
* .. Intrinsic Functions ..
INTRINSIC ABS, DBLE, DCMPLX, DIMAG, MAX, MIN
* ..
* .. Statement Functions ..
DOUBLE PRECISION CABS1
* ..
* .. Statement Function definitions ..
CABS1( ZDUM ) = ABS( DBLE( ZDUM ) ) + ABS( DIMAG( ZDUM ) )
* ..
* .. Executable Statements ..
*
RESID = ZERO
*
* Quick return if possible
*
IF( M.LE.0 .OR. N.LE.0 )
$ RETURN
*
EPS = DLAMCH( 'Precision' )
IF( M.LT.N .OR. ( M.EQ.N .AND. LSAME( ROWCOL, 'R' ) ) ) THEN
TRANSU = 'N'
K = N
ELSE
TRANSU = 'C'
K = M
END IF
MNMIN = MIN( M, N )
*
IF( ( MNMIN+1 )*MNMIN.LE.LWORK ) THEN
LDWORK = MNMIN
ELSE
LDWORK = 0
END IF
IF( LDWORK.GT.0 ) THEN
*
* Compute I - U*U' or I - U'*U.
*
CALL ZLASET( 'Upper', MNMIN, MNMIN, DCMPLX( ZERO ),
$ DCMPLX( ONE ), WORK, LDWORK )
CALL ZHERK( 'Upper', TRANSU, MNMIN, K, -ONE, U, LDU, ONE, WORK,
$ LDWORK )
*
* Compute norm( I - U*U' ) / ( K * EPS ) .
*
RESID = ZLANSY( '1', 'Upper', MNMIN, WORK, LDWORK, RWORK )
RESID = ( RESID / DBLE( K ) ) / EPS
ELSE IF( TRANSU.EQ.'C' ) THEN
*
* Find the maximum element in abs( I - U'*U ) / ( m * EPS )
*
DO 20 J = 1, N
DO 10 I = 1, J
IF( I.NE.J ) THEN
TMP = ZERO
ELSE
TMP = ONE
END IF
TMP = TMP - ZDOTC( M, U( 1, I ), 1, U( 1, J ), 1 )
RESID = MAX( RESID, CABS1( TMP ) )
10 CONTINUE
20 CONTINUE
RESID = ( RESID / DBLE( M ) ) / EPS
ELSE
*
* Find the maximum element in abs( I - U*U' ) / ( n * EPS )
*
DO 40 J = 1, M
DO 30 I = 1, J
IF( I.NE.J ) THEN
TMP = ZERO
ELSE
TMP = ONE
END IF
TMP = TMP - ZDOTC( N, U( J, 1 ), LDU, U( I, 1 ), LDU )
RESID = MAX( RESID, CABS1( TMP ) )
30 CONTINUE
40 CONTINUE
RESID = ( RESID / DBLE( N ) ) / EPS
END IF
RETURN
*
* End of ZUNT01
*
END
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?