All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.scudata.expression.fn.algebra.QRDecomposition Maven / Gradle / Ivy

Go to download

SPL(Structured Process Language) A programming language specially for structured data computing.

There is a newer version: 20240823
Show newest version
package com.scudata.expression.fn.algebra;

/**
 * ????????????Ƿֽ⴦??
 * @author bd
 */
public class QRDecomposition {
	// QR?ֽ????rows*cols?ľ?????Էֽ?Ϊ????????Q???????Ǿ???R??ˣ?rows>=cols
	// 		????QΪcols*cols?????????󣬼?Q^T*Q=Q*Q^T=I
	//		RΪrows*cols???????Ǿ?????????ֵ?????¼??·?????0
	private double[][] QR;
	private int rows, cols;
	//
	private double[] Rdiag;
	
	protected QRDecomposition(Matrix A) {
		// QR?ֽ??ʼ??.
		this.QR = A.getArrayCopy();
		this.rows = A.getRows();
		this.cols = A.getCols();
		this.Rdiag = new double[this.cols];

		// ѭ??????
		for (int k = 0; k < this.cols; k++) {
			double nrm = 0d;
			// ???㱾??
			for (int r = k; r < this.rows; r++) {
				nrm = Math.hypot(nrm, QR[r][k]);
			}
			if (nrm != 0d) {
				// Form k-th Householder vector.
				if (QR[k][k] < 0) {
					nrm = -nrm;
				}
				for (int r = k; r < this.rows; r++) {
					QR[r][k] /= nrm;
				}
				QR[k][k] += 1.0;

				// Apply transformation to remaining columns.
				for (int c = k + 1; c < this.cols; c++) {
					double s = 0d;
					for (int r = k; r < this.rows; r++) {
						s += QR[r][k] * QR[r][c];
					}
					s = -s / QR[k][k];
					for (int r = k; r < this.rows; r++) {
						QR[r][c] += s * QR[r][k];
					}
				}
			}
			Rdiag[k] = -nrm;
		}
	}
	
	/**
	 * ʹ??QR?ֽ??жϾ????Ƿ?????
	 * @return	?Ƿ?????
	 */
	protected boolean isFullRank() {
		for (int c = 0; c < this.cols; c++) {
			//if (Rdiag[c] == 0)
			if (Matrix.ifZero(this.Rdiag[c]))
				return false;
		}
		return true;
	}

	/**
	 * ??С???˷??? A*X = B
	 * @param B		???????B
	 * @return		X that minimizes the two norm of Q*R*X-B.
	 * @exception	?쳣
	 */
	protected Matrix solve(Matrix B) {
		if (B.getRows() != this.rows) {
			throw new IllegalArgumentException("Matrix row dimensions must agree.");
		}
		if (!this.isFullRank()) {
			throw new RuntimeException("Matrix is rank deficient.");
		}

		int nx = B.getCols();
		double[][] X = B.getArrayCopy();
		for (int k = 0; k < this.cols; k++) {
			for (int c = 0; c < nx; c++) {
				double s = 0d;
				for (int r = k; r < this.rows; r++) {
					s += QR[r][k] * X[r][c];
				}
				s = -s / QR[k][k];
				for (int r = k; r < this.rows; r++) {
					X[r][c] += s * QR[r][k];
				}
			}
		}
		for (int k = this.cols - 1; k >= 0; k--) {
			for (int c = 0; c < nx; c++) {
				X[k][c] /= Rdiag[k];
			}
			for (int r = 0; r < k; r++) {
				for (int c = 0; c < nx; c++) {
					X[r][c] -= X[k][c] * QR[r][k];
				}
			}
		}
		return (new Matrix(X, this.cols, nx).getMatrix(0, this.cols - 1, 0, nx - 1));
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy