Package org.apache.mahout.math

Source Code of org.apache.mahout.math.TestSingularValueDecomposition

/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements.  See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.mahout.math;

import com.google.common.base.Charsets;
import com.google.common.base.Splitter;
import com.google.common.collect.Iterables;
import com.google.common.io.Resources;
import org.apache.mahout.common.RandomUtils;
import org.apache.mahout.math.function.Functions;
import org.junit.Test;

import java.io.IOException;
import java.util.List;
import java.util.Random;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeoutException;

//To launch this test only : mvn test -Dtest=org.apache.mahout.math.TestSingularValueDecomposition
public final class TestSingularValueDecomposition extends MahoutTestCase {
 
  private final double[][] testSquare = {
      { 24.0 / 25.0, 43.0 / 25.0 },
      { 57.0 / 25.0, 24.0 / 25.0 }
  };
 
  private final double[][] testNonSquare = {
      {  -540.0 / 625.0963.0 / 625.0, -216.0 / 625.0 },
      { -1730.0 / 625.0, -744.0 / 625.0, 1008.0 / 625.0 },
      -720.0 / 625.0, 1284.0 / 625.0, -288.0 / 625.0 },
      -360.0 / 625.0192.0 / 625.0, 1756.0 / 625.0 },
  };
 
  private static final double NORM_TOLERANCE = 10.0e-14;
 
  @Test
  public void testMoreRows() {
    double[] singularValues = { 123.456, 2.3, 1.001, 0.999 };
    int rows    = singularValues.length + 2;
    int columns = singularValues.length;
    Random r = RandomUtils.getRandom();
    SingularValueDecomposition svd =
      new SingularValueDecomposition(createTestMatrix(r, rows, columns, singularValues));
    double[] computedSV = svd.getSingularValues();
    assertEquals(singularValues.length, computedSV.length);
    for (int i = 0; i < singularValues.length; ++i) {
      assertEquals(singularValues[i], computedSV[i], 1.0e-10);
    }
  }
 
  @Test
  public void testMoreColumns() {
    double[] singularValues = { 123.456, 2.3, 1.001, 0.999 };
    int rows    = singularValues.length;
    int columns = singularValues.length + 2;
    Random r = RandomUtils.getRandom();
    SingularValueDecomposition svd =
      new SingularValueDecomposition(createTestMatrix(r, rows, columns, singularValues));
    double[] computedSV = svd.getSingularValues();
    assertEquals(singularValues.length, computedSV.length);
    for (int i = 0; i < singularValues.length; ++i) {
      assertEquals(singularValues[i], computedSV[i], 1.0e-10);
    }
  }
 
  /** test dimensions */
  @Test
  public void testDimensions() {
    Matrix matrix = new DenseMatrix(testSquare);
    int m = matrix.numRows();
    int n = matrix.numCols();
    SingularValueDecomposition svd = new SingularValueDecomposition(matrix);
    assertEquals(m, svd.getU().numRows());
    assertEquals(m, svd.getU().numCols());
    assertEquals(m, svd.getS().numCols());
    assertEquals(n, svd.getS().numCols());
    assertEquals(n, svd.getV().numRows());
    assertEquals(n, svd.getV().numCols());
   
  }
 
  /** Test based on a dimension 4 Hadamard matrix. */
  // getCovariance to be implemented
  @Test
  public void testHadamard() {
    Matrix matrix = new DenseMatrix(new double[][] {
        {15.0 / 2.05.0 / 2.09.0 / 2.03.0 / 2.0 },
        { 5.0 / 2.0, 15.0 / 2.03.0 / 2.09.0 / 2.0 },
        { 9.0 / 2.03.0 / 2.0, 15.0 / 2.05.0 / 2.0 },
        { 3.0 / 2.09.0 / 2.05.0 / 2.0, 15.0 / 2.0 }
    });
    SingularValueDecomposition svd = new SingularValueDecomposition(matrix);
    assertEquals(16.0, svd.getSingularValues()[0], 1.0e-14);
    assertEquals( 8.0, svd.getSingularValues()[1], 1.0e-14);
    assertEquals( 4.0, svd.getSingularValues()[2], 1.0e-14);
    assertEquals( 2.0, svd.getSingularValues()[3], 1.0e-14);
   
    Matrix fullCovariance = new DenseMatrix(new double[][] {
        {  85.0 / 1024, -51.0 / 1024, -75.0 / 102445.0 / 1024 },
        { -51.0 / 102485.0 / 102445.0 / 1024, -75.0 / 1024 },
        { -75.0 / 102445.0 / 102485.0 / 1024, -51.0 / 1024 },
        45.0 / 1024, -75.0 / 1024, -51.0 / 102485.0 / 1024 }
    });
   
    assertEquals(0.0,Algebra.getNorm(fullCovariance.minus(svd.getCovariance(0.0))),1.0e-14);
   
   
    Matrix halfCovariance = new DenseMatrix(new double[][] {
        {   5.0 / 1024,  -3.0 / 1024,   5.0 / 1024,  -3.0 / 1024 },
        -3.0 / 1024,   5.0 / 1024,  -3.0 / 1024,   5.0 / 1024 },
        {   5.0 / 1024,  -3.0 / 1024,   5.0 / 1024,  -3.0 / 1024 },
        -3.0 / 1024,   5.0 / 1024,  -3.0 / 1024,   5.0 / 1024 }
    });
    assertEquals(0.0,Algebra.getNorm(halfCovariance.minus(svd.getCovariance(6.0))),1.0e-14);
   
  }
 
  /** test A = USVt */
  @Test
  public void testAEqualUSVt() {
    checkAEqualUSVt(new DenseMatrix(testSquare));
    checkAEqualUSVt(new DenseMatrix(testNonSquare));
    checkAEqualUSVt(new DenseMatrix(testNonSquare).transpose());
  }
 
  public static void checkAEqualUSVt(Matrix matrix) {
    SingularValueDecomposition svd = new SingularValueDecomposition(matrix);
    Matrix u = svd.getU();
    Matrix s = svd.getS();
    Matrix v = svd.getV();
   
    //pad with 0, to be able to check some properties if some singular values are equal to 0
    if (s.numRows()<matrix.numRows()) {
     
      Matrix sp = new DenseMatrix(s.numRows()+1,s.numCols());
      Matrix up = new DenseMatrix(u.numRows(),u.numCols()+1);
     
     
      for (int i = 0; i < u.numRows(); i++) {
        for (int j = 0; j < u.numCols(); j++) {
          up.set(i, j, u.get(i, j));
        }
      }

      for (int i = 0; i < s.numRows(); i++) {
        for (int j = 0; j < s.numCols(); j++) {
          sp.set(i, j, s.get(i, j));
        }
      }

      u = up;
      s = sp;
    }
   
    double norm = Algebra.getNorm(u.times(s).times(v.transpose()).minus(matrix));
    assertEquals(0, norm, NORM_TOLERANCE);
   
  }
 
  /** test that U is orthogonal */
  @Test
  public void testUOrthogonal() {
    checkOrthogonal(new SingularValueDecomposition(new DenseMatrix(testSquare)).getU());
    checkOrthogonal(new SingularValueDecomposition(new DenseMatrix(testNonSquare)).getU());
    checkOrthogonal(new SingularValueDecomposition(new DenseMatrix(testNonSquare).transpose()).getU());
  }
 
  /** test that V is orthogonal */
  @Test
  public void testVOrthogonal() {
    checkOrthogonal(new SingularValueDecomposition(new DenseMatrix(testSquare)).getV());
    checkOrthogonal(new SingularValueDecomposition(new DenseMatrix(testNonSquare)).getV());
    checkOrthogonal(new SingularValueDecomposition(new DenseMatrix(testNonSquare).transpose()).getV());
  }
 
  public static void checkOrthogonal(Matrix m) {
    Matrix mTm = m.transpose().times(m);
    Matrix id  = new DenseMatrix(mTm.numRows(),mTm.numRows());
    for (int i = 0; i < mTm.numRows(); i++) {
      id.set(i, i, 1);
    }
    assertEquals(0, Algebra.getNorm(mTm.minus(id)), NORM_TOLERANCE);
  }
 
  /** test matrices values */
  @Test
  public void testMatricesValues1() {
    SingularValueDecomposition svd =
      new SingularValueDecomposition(new DenseMatrix(testSquare));
    Matrix uRef = new DenseMatrix(new double[][] {
        { 3.0 / 5.0, 4.0 / 5.0 },
        { 4.0 / 5.0,  -3.0 / 5.0 }
    });
    Matrix sRef = new DenseMatrix(new double[][] {
        { 3.0, 0.0 },
        { 0.0, 1.0 }
    });
    Matrix vRef = new DenseMatrix(new double[][] {
        { 4.0 / 5.0,  -3.0 / 5.0 },
        { 3.0 / 5.0, 4.0 / 5.0 }
    });
   
    // check values against known references
    Matrix u = svd.getU();
   
    assertEquals(0,  Algebra.getNorm(u.minus(uRef)), NORM_TOLERANCE);
    Matrix s = svd.getS();
    assertEquals(0,  Algebra.getNorm(s.minus(sRef)), NORM_TOLERANCE);
    Matrix v = svd.getV();
    assertEquals(0,  Algebra.getNorm(v.minus(vRef)), NORM_TOLERANCE);
  }
 
 
  /** test condition number */
  @Test
  public void testConditionNumber() {
    SingularValueDecomposition svd =
      new SingularValueDecomposition(new DenseMatrix(testSquare));
    // replace 1.0e-15 with 1.5e-15
    assertEquals(3.0, svd.cond(), 1.5e-15);
  }

  @Test
  public void testSvdHang() throws IOException, InterruptedException, ExecutionException, TimeoutException {
    System.out.printf("starting hanging-svd\n");
    final Matrix m = readTsv("hanging-svd.tsv");
    SingularValueDecomposition svd = new SingularValueDecomposition(m);
    assertEquals(0, m.minus(svd.getU().times(svd.getS()).times(svd.getV().transpose())).aggregate(Functions.PLUS, Functions.ABS), 1e-10);
    System.out.printf("No hang\n");
  }

  Matrix readTsv(String name) throws IOException {
    Splitter onTab = Splitter.on("\t");
    List<String> lines = Resources.readLines((Resources.getResource(name)), Charsets.UTF_8);
    int rows = lines.size();
    int columns = Iterables.size(onTab.split(lines.get(0)));
    Matrix r = new DenseMatrix(rows, columns);
    int row = 0;
    for (String line : lines) {
      Iterable<String> values = onTab.split(line);
      int column = 0;
      for (String value : values) {
        r.set(row, column, Double.parseDouble(value));
        column++;
      }
      row++;
    }
    return r;
  }

 
  private static Matrix createTestMatrix(Random r, int rows, int columns, double[] singularValues) {
    Matrix u = createOrthogonalMatrix(r, rows);
    Matrix d = createDiagonalMatrix(singularValues, rows, columns);
    Matrix v = createOrthogonalMatrix(r, columns);
    return u.times(d).times(v);
  }
 
 
  public static Matrix createOrthogonalMatrix(Random r, int size) {
   
    double[][] data = new double[size][size];
   
    for (int i = 0; i < size; ++i) {
      double[] dataI = data[i];
      double norm2;
      do {
       
        // generate randomly row I
        for (int j = 0; j < size; ++j) {
          dataI[j] = 2 * r.nextDouble() - 1;
        }
       
        // project the row in the subspace orthogonal to previous rows
        for (int k = 0; k < i; ++k) {
          double[] dataK = data[k];
          double dotProduct = 0;
          for (int j = 0; j < size; ++j) {
            dotProduct += dataI[j] * dataK[j];
          }
          for (int j = 0; j < size; ++j) {
            dataI[j] -= dotProduct * dataK[j];
          }
        }
       
        // normalize the row
        norm2 = 0;
        for (double dataIJ : dataI) {
          norm2 += dataIJ * dataIJ;
        }
        double inv = 1.0 / Math.sqrt(norm2);
        for (int j = 0; j < size; ++j) {
          dataI[j] *= inv;
        }
       
      } while (norm2 * size < 0.01);
    }
   
    return new DenseMatrix(data);
   
  }
 
  public static Matrix createDiagonalMatrix(double[] diagonal, int rows, int columns) {
    double[][] dData = new double[rows][columns];
    for (int i = 0; i < Math.min(rows, columns); ++i) {
      dData[i][i] = diagonal[i];
    }
    return new DenseMatrix(dData);
  }
 
 
}
TOP

Related Classes of org.apache.mahout.math.TestSingularValueDecomposition

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.