org.ejml.dense.row.decomposition.qr.QRDecompositionHouseholderTran_DDRM Maven / Gradle / Ivy
/*
* Copyright (c) 2009-2018, Peter Abeles. All Rights Reserved.
*
* This file is part of Efficient Java Matrix Library (EJML).
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.ejml.dense.row.decomposition.qr;
import org.ejml.data.DMatrixRMaj;
import org.ejml.dense.row.CommonOps_DDRM;
import org.ejml.interfaces.decomposition.QRDecomposition;
/**
*
* Householder QR decomposition is rich in operations along the columns of the matrix. This can be
* taken advantage of by solving for the Q matrix in a column major format to reduce the number
* of CPU cache misses and the number of copies that are performed.
*
*
* @see QRDecompositionHouseholder_DDRM
*
* @author Peter Abeles
*/
// TODO remove QR Col and replace with this one?
// -- On small matrices col seems to be about 10% faster
public class QRDecompositionHouseholderTran_DDRM implements QRDecomposition {
/**
* Where the Q and R matrices are stored. For speed reasons
* this is transposed
*/
protected DMatrixRMaj QR;
// used internally to store temporary data
protected double v[];
// dimension of the decomposed matrices
protected int numCols; // this is 'n'
protected int numRows; // this is 'm'
protected int minLength;
// the computed gamma for Q_k matrix
protected double gammas[];
// local variables
protected double gamma;
protected double tau;
// did it encounter an error?
protected boolean error;
public void setExpectedMaxSize( int numRows , int numCols ) {
this.numCols = numCols;
this.numRows = numRows;
minLength = Math.min(numCols,numRows);
int maxLength = Math.max(numCols,numRows);
if( QR == null ) {
QR = new DMatrixRMaj(numCols,numRows);
v = new double[ maxLength ];
gammas = new double[ minLength ];
} else {
QR.reshape(numCols,numRows,false);
}
if( v.length < maxLength ) {
v = new double[ maxLength ];
}
if( gammas.length < minLength ) {
gammas = new double[ minLength ];
}
}
/**
* Inner matrix that stores the decomposition
*/
public DMatrixRMaj getQR() {
return QR;
}
/**
* Computes the Q matrix from the information stored in the QR matrix. This
* operation requires about 4(m2n-mn2+n3/3) flops.
*
* @param Q The orthogonal Q matrix.
*/
@Override
public DMatrixRMaj getQ(DMatrixRMaj Q , boolean compact ) {
if( compact ) {
if( Q == null ) {
Q = CommonOps_DDRM.identity(numRows,minLength);
} else {
if( Q.numRows != numRows || Q.numCols != minLength ) {
throw new IllegalArgumentException("Unexpected matrix dimension.");
} else {
CommonOps_DDRM.setIdentity(Q);
}
}
} else {
if( Q == null ) {
Q = CommonOps_DDRM.identity(numRows);
} else {
if( Q.numRows != numRows || Q.numCols != numRows ) {
throw new IllegalArgumentException("Unexpected matrix dimension.");
} else {
CommonOps_DDRM.setIdentity(Q);
}
}
}
// Unlike applyQ() this takes advantage of zeros in the identity matrix
// by not multiplying across all rows.
for( int j = minLength-1; j >= 0; j-- ) {
int diagIndex = j*numRows+j;
double before = QR.data[diagIndex];
QR.data[diagIndex] = 1;
QrHelperFunctions_DDRM.rank1UpdateMultR(Q, QR.data, j * numRows, gammas[j], j, j, numRows, v);
QR.data[diagIndex] = before;
}
return Q;
}
/**
* A = Q*A
*
* @param A Matrix that is being multiplied by Q. Is modified.
*/
public void applyQ( DMatrixRMaj A ) {
if( A.numRows != numRows )
throw new IllegalArgumentException("A must have at least "+numRows+" rows.");
for( int j = minLength-1; j >= 0; j-- ) {
int diagIndex = j*numRows+j;
double before = QR.data[diagIndex];
QR.data[diagIndex] = 1;
QrHelperFunctions_DDRM.rank1UpdateMultR(A, QR.data, j * numRows, gammas[j], 0, j, numRows, v);
QR.data[diagIndex] = before;
}
}
/**
* A = QT*A
*
* @param A Matrix that is being multiplied by QT. Is modified.
*/
public void applyTranQ( DMatrixRMaj A ) {
for( int j = 0; j < minLength; j++ ) {
int diagIndex = j*numRows+j;
double before = QR.data[diagIndex];
QR.data[diagIndex] = 1;
QrHelperFunctions_DDRM.rank1UpdateMultR(A, QR.data, j * numRows, gammas[j], 0, j, numRows, v);
QR.data[diagIndex] = before;
}
}
/**
* Returns an upper triangular matrix which is the R in the QR decomposition.
*
* @param R An upper triangular matrix.
* @param compact
*/
@Override
public DMatrixRMaj getR(DMatrixRMaj R, boolean compact) {
if( R == null ) {
if( compact ) {
R = new DMatrixRMaj(minLength,numCols);
} else
R = new DMatrixRMaj(numRows,numCols);
} else {
if( compact ) {
R.reshape(minLength,numCols);
} else {
R.reshape(numRows,numCols);
}
for( int i = 0; i < R.numRows; i++ ) {
int min = Math.min(i,R.numCols);
for( int j = 0; j < min; j++ ) {
R.unsafe_set(i,j,0);
}
}
}
for( int i = 0; i < R.numRows; i++ ) {
for( int j = i; j < R.numCols; j++ ) {
R.unsafe_set(i,j,QR.unsafe_get(j,i));
}
}
return R;
}
/**
*
* To decompose the matrix 'A' it must have full rank. 'A' is a 'm' by 'n' matrix.
* It requires about 2n*m2-2m2/3 flops.
*
*
*
* The matrix provided here can be of different
* dimension than the one specified in the constructor. It just has to be smaller than or equal
* to it.
*
*/
@Override
public boolean decompose( DMatrixRMaj A ) {
setExpectedMaxSize(A.numRows, A.numCols);
CommonOps_DDRM.transpose(A,QR);
error = false;
for( int j = 0; j < minLength; j++ ) {
householder(j);
updateA(j);
}
return !error;
}
@Override
public boolean inputModified() {
return false;
}
/**
*
* Computes the householder vector "u" for the first column of submatrix j. Note this is
* a specialized householder for this problem. There is some protection against
* overflow and underflow.
*
*
* Q = I - γuuT
*
*
* This function finds the values of 'u' and 'γ'.
*
*
* @param j Which submatrix to work off of.
*/
protected void householder( final int j )
{
int startQR = j*numRows;
int endQR = startQR+numRows;
startQR += j;
final double max = QrHelperFunctions_DDRM.findMax(QR.data, startQR, numRows - j);
if( max == 0.0 ) {
gamma = 0;
error = true;
} else {
// computes tau and normalizes u by max
tau = QrHelperFunctions_DDRM.computeTauAndDivide(startQR, endQR, QR.data, max);
// divide u by u_0
double u_0 = QR.data[startQR] + tau;
QrHelperFunctions_DDRM.divideElements(startQR + 1, endQR, QR.data, u_0);
gamma = u_0/tau;
tau *= max;
QR.data[startQR] = -tau;
}
gammas[j] = gamma;
}
/**
*
* Takes the results from the householder computation and updates the 'A' matrix.
*
* A = (I - γ*u*uT)A
*
*
* @param w The submatrix.
*/
protected void updateA( final int w )
{
// int rowW = w*numRows;
// int rowJ = rowW + numRows;
//
// for( int j = w+1; j < numCols; j++ , rowJ += numRows) {
// double val = QR.data[rowJ + w];
//
// // val = gamma*u^T * A
// for( int k = w+1; k < numRows; k++ ) {
// val += QR.data[rowW + k]*QR.data[rowJ + k];
// }
// val *= gamma;
//
// // A - val*u
// QR.data[rowJ + w] -= val;
// for( int i = w+1; i < numRows; i++ ) {
// QR.data[rowJ + i] -= QR.data[rowW + i]*val;
// }
// }
final double data[] = QR.data;
final int rowW = w*numRows + w + 1;
int rowJ = rowW + numRows;
final int rowJEnd = rowJ + (numCols-w-1)*numRows;
final int indexWEnd = rowW + numRows - w - 1;
for( ; rowJEnd != rowJ; rowJ += numRows) {
// assume the first element in u is 1
double val = data[rowJ - 1];
int indexW = rowW;
int indexJ = rowJ;
while( indexW != indexWEnd ) {
val += data[indexW++]*data[indexJ++];
}
val *= gamma;
data[rowJ - 1] -= val;
indexW = rowW;
indexJ = rowJ;
while( indexW != indexWEnd ) {
data[indexJ++] -= data[indexW++]*val;
}
}
}
public double[] getGammas() {
return gammas;
}
}