/*
* Copyright 2010, 2011 Institut Pasteur.
*
* This file is part of NHerve Main Toolbox, which is an ICY plugin.
*
* NHerve Main Toolbox is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* NHerve Main Toolbox is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with NHerve Main Toolbox. If not, see <http://www.gnu.org/licenses/>.
*/
package plugins.nherve.toolbox.image.feature;
import java.util.ArrayList;
import java.util.List;
import plugins.nherve.matrix.EigenvalueDecomposition;
import plugins.nherve.matrix.Matrix;
import plugins.nherve.toolbox.image.feature.signature.DenseVectorSignature;
import plugins.nherve.toolbox.image.feature.signature.SignatureException;
import plugins.nherve.toolbox.image.feature.signature.DefaultVectorSignature;
/**
* The Class PCA.
*
* @author Nicolas HERVE - nicolas.herve@pasteur.fr
*/
public class PCA extends DimensionReductionAlgorithm {
/** The evd. */
private EigenvalueDecomposition evd;
/**
* Instantiates a new pCA.
*
* @param signatures
* the signatures
*/
public PCA(List<DefaultVectorSignature> signatures) {
super(signatures);
}
/* (non-Javadoc)
* @see plugins.nherve.toolbox.image.feature.DimensionReductionAlgorithm#compute()
*/
@Override
public void compute() throws SignatureException {
check();
Matrix m = getCenteredMatrix(signatures);
log("Building variance/covariance matrix ...");
Matrix varcov = getVarCovMatrix(m);
m = null;
log("Launching EigenvalueDecomposition ... " + varcov.getRowDimension() + " x " + varcov.getColumnDimension());
evd = varcov.eig();
if (isLogEnabled()) {
evd.getV().print(8, 3);
}
log("Done");
}
/**
* Gets the projection matrix.
*
* @return the projection matrix
*/
public Matrix getProjectionMatrix() {
return evd.getV();
}
/* (non-Javadoc)
* @see plugins.nherve.toolbox.image.feature.DimensionReductionAlgorithm#project(java.util.List)
*/
@Override
public List<DefaultVectorSignature> project(List<DefaultVectorSignature> toProject) throws SignatureException {
Matrix tp = getCenteredMatrix(toProject);
Matrix p = tp.times(evd.getV());
ArrayList<DefaultVectorSignature> proj = new ArrayList<DefaultVectorSignature>();
for (int s = 0; s < toProject.size(); s++) {
DenseVectorSignature vs = new DenseVectorSignature(p.getColumnDimension());
for (int d = 0; d < p.getColumnDimension(); d++) {
vs.set(d, p.get(s, d));
}
proj.add(vs);
}
return proj;
}
/**
* Project.
*
* @param toProject
* the to project
* @param upToDim
* the up to dim
* @return the list
* @throws SignatureException
* the signature exception
*/
public List<DefaultVectorSignature> project(List<DefaultVectorSignature> toProject, int upToDim) throws SignatureException {
Matrix tp = getCenteredMatrix(toProject);
Matrix p = tp.times(evd.getV());
if (upToDim > p.getColumnDimension()) {
throw new SignatureException("Can not project up to dimension " + upToDim + " / " + p.getColumnDimension());
}
ArrayList<DefaultVectorSignature> proj = new ArrayList<DefaultVectorSignature>();
for (int s = 0; s < toProject.size(); s++) {
DenseVectorSignature vs = new DenseVectorSignature(upToDim);
for (int d = 0; d < upToDim; d++) {
vs.set(d, p.get(s, d));
}
vs.normalizeSumToOne(true);
proj.add(vs);
}
return proj;
}
}