MatrixUtils.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      https://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */

  17. /*
  18.  * This is not the original file distributed by the Apache Software Foundation
  19.  * It has been modified by the Hipparchus project
  20.  */

  21. package org.hipparchus.linear;

  22. import java.util.ArrayList;
  23. import java.util.Arrays;
  24. import java.util.List;

  25. import org.hipparchus.CalculusFieldElement;
  26. import org.hipparchus.Field;
  27. import org.hipparchus.FieldElement;
  28. import org.hipparchus.exception.LocalizedCoreFormats;
  29. import org.hipparchus.exception.MathIllegalArgumentException;
  30. import org.hipparchus.exception.MathRuntimeException;
  31. import org.hipparchus.exception.NullArgumentException;
  32. import org.hipparchus.fraction.BigFraction;
  33. import org.hipparchus.fraction.Fraction;
  34. import org.hipparchus.util.FastMath;
  35. import org.hipparchus.util.MathArrays;
  36. import org.hipparchus.util.MathUtils;
  37. import org.hipparchus.util.Precision;

  38. /**
  39.  * A collection of static methods that operate on or return matrices.
  40.  *
  41.  */
  42. public class MatrixUtils {

  43.     /**
  44.      * The default format for {@link RealMatrix} objects.
  45.      */
  46.     public static final RealMatrixFormat DEFAULT_FORMAT = RealMatrixFormat.getRealMatrixFormat();

  47.     /**
  48.      * A format for {@link RealMatrix} objects compatible with octave.
  49.      */
  50.     public static final RealMatrixFormat OCTAVE_FORMAT = new RealMatrixFormat("[", "]", "", "", "; ", ", ");

  51.     /** Pade coefficients required for the matrix exponential calculation. */
  52.     private static final double[] PADE_COEFFICIENTS_3 = {
  53.             120.0,
  54.             60.0,
  55.             12.0,
  56.             1.0
  57.     };

  58.     /** Pade coefficients required for the matrix exponential calculation. */
  59.     private static final double[] PADE_COEFFICIENTS_5 = {
  60.             30240.0,
  61.             15120.0,
  62.             3360.0,
  63.             420.0,
  64.             30.0,
  65.             1
  66.     };

  67.     /** Pade coefficients required for the matrix exponential calculation. */
  68.     private static final double[] PADE_COEFFICIENTS_7 = {
  69.             17297280.0,
  70.             8648640.0,
  71.             1995840.0,
  72.             277200.0,
  73.             25200.0,
  74.             1512.0,
  75.             56.0,
  76.             1.0
  77.     };

  78.     /** Pade coefficients required for the matrix exponential calculation. */
  79.     private static final double[] PADE_COEFFICIENTS_9 = {
  80.             17643225600.0,
  81.             8821612800.0,
  82.             2075673600.0,
  83.             302702400.0,
  84.             30270240.0,
  85.             2162160.0,
  86.             110880.0,
  87.             3960.0,
  88.             90.0,
  89.             1.0
  90.     };

  91.     /** Pade coefficients required for the matrix exponential calculation. */
  92.     private static final double[] PADE_COEFFICIENTS_13 = {
  93.             6.476475253248e+16,
  94.             3.238237626624e+16,
  95.             7.7717703038976e+15,
  96.             1.1873537964288e+15,
  97.             129060195264000.0,
  98.             10559470521600.0,
  99.             670442572800.0,
  100.             33522128640.0,
  101.             1323241920.0,
  102.             40840800.0,
  103.             960960.0,
  104.             16380.0,
  105.             182.0,
  106.             1.0
  107.     };

  108.     /**
  109.      * Private constructor.
  110.      */
  111.     private MatrixUtils() {
  112.         super();
  113.     }

  114.     /**
  115.      * Returns a {@link RealMatrix} with specified dimensions.
  116.      * <p>The type of matrix returned depends on the dimension. Below
  117.      * 2<sup>12</sup> elements (i.e. 4096 elements or 64&times;64 for a
  118.      * square matrix) which can be stored in a 32kB array, a {@link
  119.      * Array2DRowRealMatrix} instance is built. Above this threshold a {@link
  120.      * BlockRealMatrix} instance is built.</p>
  121.      * <p>The matrix elements are all set to 0.0.</p>
  122.      * @param rows number of rows of the matrix
  123.      * @param columns number of columns of the matrix
  124.      * @return  RealMatrix with specified dimensions
  125.      * @see #createRealMatrix(double[][])
  126.      */
  127.     public static RealMatrix createRealMatrix(final int rows, final int columns) {
  128.         return (rows * columns <= 4096) ?
  129.                 new Array2DRowRealMatrix(rows, columns) : new BlockRealMatrix(rows, columns);
  130.     }

  131.     /**
  132.      * Returns a {@link FieldMatrix} with specified dimensions.
  133.      * <p>The type of matrix returned depends on the dimension. Below
  134.      * 2<sup>12</sup> elements (i.e. 4096 elements or 64&times;64 for a
  135.      * square matrix), a {@link FieldMatrix} instance is built. Above
  136.      * this threshold a {@link BlockFieldMatrix} instance is built.</p>
  137.      * <p>The matrix elements are all set to field.getZero().</p>
  138.      * @param <T> the type of the field elements
  139.      * @param field field to which the matrix elements belong
  140.      * @param rows number of rows of the matrix
  141.      * @param columns number of columns of the matrix
  142.      * @return  FieldMatrix with specified dimensions
  143.      * @see #createFieldMatrix(FieldElement[][])
  144.      */
  145.     public static <T extends FieldElement<T>> FieldMatrix<T> createFieldMatrix(final Field<T> field,
  146.                                                                                final int rows,
  147.                                                                                final int columns) {
  148.         return (rows * columns <= 4096) ?
  149.                 new Array2DRowFieldMatrix<>(field, rows, columns) : new BlockFieldMatrix<>(field, rows, columns);
  150.     }

  151.     /**
  152.      * Returns a {@link RealMatrix} whose entries are the the values in the
  153.      * the input array.
  154.      * <p>The type of matrix returned depends on the dimension. Below
  155.      * 2<sup>12</sup> elements (i.e. 4096 elements or 64&times;64 for a
  156.      * square matrix) which can be stored in a 32kB array, a {@link
  157.      * Array2DRowRealMatrix} instance is built. Above this threshold a {@link
  158.      * BlockRealMatrix} instance is built.</p>
  159.      * <p>The input array is copied, not referenced.</p>
  160.      *
  161.      * @param data input array
  162.      * @return  RealMatrix containing the values of the array
  163.      * @throws org.hipparchus.exception.MathIllegalArgumentException
  164.      * if {@code data} is not rectangular (not all rows have the same length).
  165.      * @throws MathIllegalArgumentException if a row or column is empty.
  166.      * @throws NullArgumentException if either {@code data} or {@code data[0]}
  167.      * is {@code null}.
  168.      * @throws MathIllegalArgumentException if {@code data} is not rectangular.
  169.      * @see #createRealMatrix(int, int)
  170.      */
  171.     public static RealMatrix createRealMatrix(double[][] data)
  172.         throws MathIllegalArgumentException, NullArgumentException {
  173.         if (data == null ||
  174.             data[0] == null) {
  175.             throw new NullArgumentException();
  176.         }
  177.         return (data.length * data[0].length <= 4096) ?
  178.                 new Array2DRowRealMatrix(data) : new BlockRealMatrix(data);
  179.     }

  180.     /**
  181.      * Returns a {@link FieldMatrix} whose entries are the the values in the
  182.      * the input array.
  183.      * <p>The type of matrix returned depends on the dimension. Below
  184.      * 2<sup>12</sup> elements (i.e. 4096 elements or 64&times;64 for a
  185.      * square matrix), a {@link FieldMatrix} instance is built. Above
  186.      * this threshold a {@link BlockFieldMatrix} instance is built.</p>
  187.      * <p>The input array is copied, not referenced.</p>
  188.      * @param <T> the type of the field elements
  189.      * @param data input array
  190.      * @return a matrix containing the values of the array.
  191.      * @throws org.hipparchus.exception.MathIllegalArgumentException
  192.      * if {@code data} is not rectangular (not all rows have the same length).
  193.      * @throws MathIllegalArgumentException if a row or column is empty.
  194.      * @throws NullArgumentException if either {@code data} or {@code data[0]}
  195.      * is {@code null}.
  196.      * @see #createFieldMatrix(Field, int, int)
  197.      */
  198.     public static <T extends FieldElement<T>> FieldMatrix<T> createFieldMatrix(T[][] data)
  199.         throws MathIllegalArgumentException, NullArgumentException {
  200.         if (data == null ||
  201.             data[0] == null) {
  202.             throw new NullArgumentException();
  203.         }
  204.         return (data.length * data[0].length <= 4096) ?
  205.                 new Array2DRowFieldMatrix<>(data) : new BlockFieldMatrix<>(data);
  206.     }

  207.     /**
  208.      * Returns <code>dimension x dimension</code> identity matrix.
  209.      *
  210.      * @param dimension dimension of identity matrix to generate
  211.      * @return identity matrix
  212.      * @throws IllegalArgumentException if dimension is not positive
  213.      */
  214.     public static RealMatrix createRealIdentityMatrix(int dimension) {
  215.         final RealMatrix m = createRealMatrix(dimension, dimension);
  216.         for (int i = 0; i < dimension; ++i) {
  217.             m.setEntry(i, i, 1.0);
  218.         }
  219.         return m;
  220.     }

  221.     /**
  222.      * Returns <code>dimension x dimension</code> identity matrix.
  223.      *
  224.      * @param <T> the type of the field elements
  225.      * @param field field to which the elements belong
  226.      * @param dimension dimension of identity matrix to generate
  227.      * @return identity matrix
  228.      * @throws IllegalArgumentException if dimension is not positive
  229.      */
  230.     public static <T extends FieldElement<T>> FieldMatrix<T>
  231.         createFieldIdentityMatrix(final Field<T> field, final int dimension) {
  232.         final T zero = field.getZero();
  233.         final T one  = field.getOne();
  234.         final T[][] d = MathArrays.buildArray(field, dimension, dimension);
  235.         for (int row = 0; row < dimension; row++) {
  236.             final T[] dRow = d[row];
  237.             Arrays.fill(dRow, zero);
  238.             dRow[row] = one;
  239.         }
  240.         return new Array2DRowFieldMatrix<>(field, d, false);
  241.     }

  242.     /**
  243.      * Returns a diagonal matrix with specified elements.
  244.      *
  245.      * @param diagonal diagonal elements of the matrix (the array elements
  246.      * will be copied)
  247.      * @return diagonal matrix
  248.      */
  249.     public static RealMatrix createRealDiagonalMatrix(final double[] diagonal) {
  250.         final RealMatrix m = createRealMatrix(diagonal.length, diagonal.length);
  251.         for (int i = 0; i < diagonal.length; ++i) {
  252.             m.setEntry(i, i, diagonal[i]);
  253.         }
  254.         return m;
  255.     }

  256.     /**
  257.      * Returns a diagonal matrix with specified elements.
  258.      *
  259.      * @param <T> the type of the field elements
  260.      * @param diagonal diagonal elements of the matrix (the array elements
  261.      * will be copied)
  262.      * @return diagonal matrix
  263.      */
  264.     public static <T extends FieldElement<T>> FieldMatrix<T>
  265.         createFieldDiagonalMatrix(final T[] diagonal) {
  266.         final FieldMatrix<T> m =
  267.             createFieldMatrix(diagonal[0].getField(), diagonal.length, diagonal.length);
  268.         for (int i = 0; i < diagonal.length; ++i) {
  269.             m.setEntry(i, i, diagonal[i]);
  270.         }
  271.         return m;
  272.     }

  273.     /**
  274.      * Creates a {@link RealVector} using the data from the input array.
  275.      *
  276.      * @param data the input data
  277.      * @return a data.length RealVector
  278.      * @throws MathIllegalArgumentException if {@code data} is empty.
  279.      * @throws NullArgumentException if {@code data} is {@code null}.
  280.      */
  281.     public static RealVector createRealVector(double[] data)
  282.         throws MathIllegalArgumentException, NullArgumentException {
  283.         if (data == null) {
  284.             throw new NullArgumentException();
  285.         }
  286.         return new ArrayRealVector(data, true);
  287.     }

  288.     /**
  289.      * Creates a {@link RealVector} with specified dimensions.
  290.      *
  291.      * @param dimension dimension of the vector
  292.      * @return a new vector
  293.      * @since 1.3
  294.      */
  295.     public static RealVector createRealVector(final int dimension) {
  296.         return new ArrayRealVector(new double[dimension]);
  297.     }

  298.     /**
  299.      * Creates a {@link FieldVector} using the data from the input array.
  300.      *
  301.      * @param <T> the type of the field elements
  302.      * @param data the input data
  303.      * @return a data.length FieldVector
  304.      * @throws MathIllegalArgumentException if {@code data} is empty.
  305.      * @throws NullArgumentException if {@code data} is {@code null}.
  306.      * @throws MathIllegalArgumentException if {@code data} has 0 elements
  307.      */
  308.     public static <T extends FieldElement<T>> FieldVector<T> createFieldVector(final T[] data)
  309.         throws MathIllegalArgumentException, NullArgumentException {
  310.         if (data == null) {
  311.             throw new NullArgumentException();
  312.         }
  313.         if (data.length == 0) {
  314.             throw new MathIllegalArgumentException(LocalizedCoreFormats.VECTOR_MUST_HAVE_AT_LEAST_ONE_ELEMENT);
  315.         }
  316.         return new ArrayFieldVector<>(data[0].getField(), data, true);
  317.     }

  318.     /**
  319.      * Creates a {@link FieldVector} with specified dimensions.
  320.      *
  321.      * @param <T> the type of the field elements
  322.      * @param field field to which array elements belong
  323.      * @param dimension dimension of the vector
  324.      * @return a new vector
  325.      * @since 1.3
  326.      */
  327.     public static <T extends FieldElement<T>> FieldVector<T> createFieldVector(final Field<T> field, final int dimension) {
  328.         return new ArrayFieldVector<>(MathArrays.buildArray(field, dimension));
  329.     }

  330.     /**
  331.      * Create a row {@link RealMatrix} using the data from the input
  332.      * array.
  333.      *
  334.      * @param rowData the input row data
  335.      * @return a 1 x rowData.length RealMatrix
  336.      * @throws MathIllegalArgumentException if {@code rowData} is empty.
  337.      * @throws NullArgumentException if {@code rowData} is {@code null}.
  338.      */
  339.     public static RealMatrix createRowRealMatrix(double[] rowData)
  340.         throws MathIllegalArgumentException, NullArgumentException {
  341.         if (rowData == null) {
  342.             throw new NullArgumentException();
  343.         }
  344.         final int nCols = rowData.length;
  345.         final RealMatrix m = createRealMatrix(1, nCols);
  346.         for (int i = 0; i < nCols; ++i) {
  347.             m.setEntry(0, i, rowData[i]);
  348.         }
  349.         return m;
  350.     }

  351.     /**
  352.      * Create a row {@link FieldMatrix} using the data from the input
  353.      * array.
  354.      *
  355.      * @param <T> the type of the field elements
  356.      * @param rowData the input row data
  357.      * @return a 1 x rowData.length FieldMatrix
  358.      * @throws MathIllegalArgumentException if {@code rowData} is empty.
  359.      * @throws NullArgumentException if {@code rowData} is {@code null}.
  360.      */
  361.     public static <T extends FieldElement<T>> FieldMatrix<T>
  362.         createRowFieldMatrix(final T[] rowData)
  363.         throws MathIllegalArgumentException, NullArgumentException {
  364.         if (rowData == null) {
  365.             throw new NullArgumentException();
  366.         }
  367.         final int nCols = rowData.length;
  368.         if (nCols == 0) {
  369.             throw new MathIllegalArgumentException(LocalizedCoreFormats.AT_LEAST_ONE_COLUMN);
  370.         }
  371.         final FieldMatrix<T> m = createFieldMatrix(rowData[0].getField(), 1, nCols);
  372.         for (int i = 0; i < nCols; ++i) {
  373.             m.setEntry(0, i, rowData[i]);
  374.         }
  375.         return m;
  376.     }

  377.     /**
  378.      * Creates a column {@link RealMatrix} using the data from the input
  379.      * array.
  380.      *
  381.      * @param columnData  the input column data
  382.      * @return a columnData x 1 RealMatrix
  383.      * @throws MathIllegalArgumentException if {@code columnData} is empty.
  384.      * @throws NullArgumentException if {@code columnData} is {@code null}.
  385.      */
  386.     public static RealMatrix createColumnRealMatrix(double[] columnData)
  387.         throws MathIllegalArgumentException, NullArgumentException {
  388.         if (columnData == null) {
  389.             throw new NullArgumentException();
  390.         }
  391.         final int nRows = columnData.length;
  392.         final RealMatrix m = createRealMatrix(nRows, 1);
  393.         for (int i = 0; i < nRows; ++i) {
  394.             m.setEntry(i, 0, columnData[i]);
  395.         }
  396.         return m;
  397.     }

  398.     /**
  399.      * Creates a column {@link FieldMatrix} using the data from the input
  400.      * array.
  401.      *
  402.      * @param <T> the type of the field elements
  403.      * @param columnData  the input column data
  404.      * @return a columnData x 1 FieldMatrix
  405.      * @throws MathIllegalArgumentException if {@code data} is empty.
  406.      * @throws NullArgumentException if {@code columnData} is {@code null}.
  407.      */
  408.     public static <T extends FieldElement<T>> FieldMatrix<T>
  409.         createColumnFieldMatrix(final T[] columnData)
  410.         throws MathIllegalArgumentException, NullArgumentException {
  411.         if (columnData == null) {
  412.             throw new NullArgumentException();
  413.         }
  414.         final int nRows = columnData.length;
  415.         if (nRows == 0) {
  416.             throw new MathIllegalArgumentException(LocalizedCoreFormats.AT_LEAST_ONE_ROW);
  417.         }
  418.         final FieldMatrix<T> m = createFieldMatrix(columnData[0].getField(), nRows, 1);
  419.         for (int i = 0; i < nRows; ++i) {
  420.             m.setEntry(i, 0, columnData[i]);
  421.         }
  422.         return m;
  423.     }

  424.     /**
  425.      * Checks whether a matrix is symmetric, within a given relative tolerance.
  426.      *
  427.      * @param matrix Matrix to check.
  428.      * @param relativeTolerance Tolerance of the symmetry check.
  429.      * @param raiseException If {@code true}, an exception will be raised if
  430.      * the matrix is not symmetric.
  431.      * @return {@code true} if {@code matrix} is symmetric.
  432.      * @throws MathIllegalArgumentException if the matrix is not square.
  433.      * @throws MathIllegalArgumentException if the matrix is not symmetric.
  434.      */
  435.     private static boolean isSymmetricInternal(RealMatrix matrix,
  436.                                                double relativeTolerance,
  437.                                                boolean raiseException) {
  438.         final int rows = matrix.getRowDimension();
  439.         if (rows != matrix.getColumnDimension()) {
  440.             if (raiseException) {
  441.                 throw new MathIllegalArgumentException(LocalizedCoreFormats.NON_SQUARE_MATRIX,
  442.                                                        rows, matrix.getColumnDimension());
  443.             } else {
  444.                 return false;
  445.             }
  446.         }
  447.         for (int i = 0; i < rows; i++) {
  448.             for (int j = i + 1; j < rows; j++) {
  449.                 final double mij = matrix.getEntry(i, j);
  450.                 final double mji = matrix.getEntry(j, i);
  451.                 if (FastMath.abs(mij - mji) >
  452.                     FastMath.max(FastMath.abs(mij), FastMath.abs(mji)) * relativeTolerance) {
  453.                     if (raiseException) {
  454.                         throw new MathIllegalArgumentException(LocalizedCoreFormats.NON_SYMMETRIC_MATRIX,
  455.                                                                i, j, relativeTolerance);
  456.                     } else {
  457.                         return false;
  458.                     }
  459.                 }
  460.             }
  461.         }
  462.         return true;
  463.     }

  464.     /**
  465.      * Checks whether a matrix is symmetric.
  466.      *
  467.      * @param matrix Matrix to check.
  468.      * @param eps Relative tolerance.
  469.      * @throws MathIllegalArgumentException if the matrix is not square.
  470.      * @throws MathIllegalArgumentException if the matrix is not symmetric.
  471.      */
  472.     public static void checkSymmetric(RealMatrix matrix,
  473.                                       double eps) {
  474.         isSymmetricInternal(matrix, eps, true);
  475.     }

  476.     /**
  477.      * Checks whether a matrix is symmetric.
  478.      *
  479.      * @param matrix Matrix to check.
  480.      * @param eps Relative tolerance.
  481.      * @return {@code true} if {@code matrix} is symmetric.
  482.      */
  483.     public static boolean isSymmetric(RealMatrix matrix,
  484.                                       double eps) {
  485.         return isSymmetricInternal(matrix, eps, false);
  486.     }

  487.     /**
  488.      * Check if matrix indices are valid.
  489.      *
  490.      * @param m Matrix.
  491.      * @param row Row index to check.
  492.      * @param column Column index to check.
  493.      * @throws MathIllegalArgumentException if {@code row} or {@code column} is not
  494.      * a valid index.
  495.      */
  496.     public static void checkMatrixIndex(final AnyMatrix m,
  497.                                         final int row, final int column)
  498.         throws MathIllegalArgumentException {
  499.         checkRowIndex(m, row);
  500.         checkColumnIndex(m, column);
  501.     }

  502.     /**
  503.      * Check if a row index is valid.
  504.      *
  505.      * @param m Matrix.
  506.      * @param row Row index to check.
  507.      * @throws MathIllegalArgumentException if {@code row} is not a valid index.
  508.      */
  509.     public static void checkRowIndex(final AnyMatrix m, final int row)
  510.         throws MathIllegalArgumentException {
  511.         if (row < 0 ||
  512.             row >= m.getRowDimension()) {
  513.             throw new MathIllegalArgumentException(LocalizedCoreFormats.ROW_INDEX,
  514.                                           row, 0, m.getRowDimension() - 1);
  515.         }
  516.     }

  517.     /**
  518.      * Check if a column index is valid.
  519.      *
  520.      * @param m Matrix.
  521.      * @param column Column index to check.
  522.      * @throws MathIllegalArgumentException if {@code column} is not a valid index.
  523.      */
  524.     public static void checkColumnIndex(final AnyMatrix m, final int column)
  525.         throws MathIllegalArgumentException {
  526.         if (column < 0 || column >= m.getColumnDimension()) {
  527.             throw new MathIllegalArgumentException(LocalizedCoreFormats.COLUMN_INDEX,
  528.                                            column, 0, m.getColumnDimension() - 1);
  529.         }
  530.     }

  531.     /**
  532.      * Check if submatrix ranges indices are valid.
  533.      * Rows and columns are indicated counting from 0 to {@code n - 1}.
  534.      *
  535.      * @param m Matrix.
  536.      * @param startRow Initial row index.
  537.      * @param endRow Final row index.
  538.      * @param startColumn Initial column index.
  539.      * @param endColumn Final column index.
  540.      * @throws MathIllegalArgumentException if the indices are invalid.
  541.      * @throws MathIllegalArgumentException if {@code endRow < startRow} or
  542.      * {@code endColumn < startColumn}.
  543.      */
  544.     public static void checkSubMatrixIndex(final AnyMatrix m,
  545.                                            final int startRow, final int endRow,
  546.                                            final int startColumn, final int endColumn)
  547.         throws MathIllegalArgumentException {
  548.         checkRowIndex(m, startRow);
  549.         checkRowIndex(m, endRow);
  550.         if (endRow < startRow) {
  551.             throw new MathIllegalArgumentException(LocalizedCoreFormats.INITIAL_ROW_AFTER_FINAL_ROW,
  552.                                                 endRow, startRow, false);
  553.         }

  554.         checkColumnIndex(m, startColumn);
  555.         checkColumnIndex(m, endColumn);
  556.         if (endColumn < startColumn) {
  557.             throw new MathIllegalArgumentException(LocalizedCoreFormats.INITIAL_COLUMN_AFTER_FINAL_COLUMN,
  558.                                                 endColumn, startColumn, false);
  559.         }


  560.     }

  561.     /**
  562.      * Check if submatrix ranges indices are valid.
  563.      * Rows and columns are indicated counting from 0 to n-1.
  564.      *
  565.      * @param m Matrix.
  566.      * @param selectedRows Array of row indices.
  567.      * @param selectedColumns Array of column indices.
  568.      * @throws NullArgumentException if {@code selectedRows} or
  569.      * {@code selectedColumns} are {@code null}.
  570.      * @throws MathIllegalArgumentException if the row or column selections are empty (zero
  571.      * length).
  572.      * @throws MathIllegalArgumentException if row or column selections are not valid.
  573.      */
  574.     public static void checkSubMatrixIndex(final AnyMatrix m,
  575.                                            final int[] selectedRows,
  576.                                            final int[] selectedColumns)
  577.         throws MathIllegalArgumentException, NullArgumentException {
  578.         if (selectedRows == null) {
  579.             throw new NullArgumentException();
  580.         }
  581.         if (selectedColumns == null) {
  582.             throw new NullArgumentException();
  583.         }
  584.         if (selectedRows.length == 0) {
  585.             throw new MathIllegalArgumentException(LocalizedCoreFormats.EMPTY_SELECTED_ROW_INDEX_ARRAY);
  586.         }
  587.         if (selectedColumns.length == 0) {
  588.             throw new MathIllegalArgumentException(LocalizedCoreFormats.EMPTY_SELECTED_COLUMN_INDEX_ARRAY);
  589.         }

  590.         for (final int row : selectedRows) {
  591.             checkRowIndex(m, row);
  592.         }
  593.         for (final int column : selectedColumns) {
  594.             checkColumnIndex(m, column);
  595.         }
  596.     }

  597.     /**
  598.      * Check if matrices are addition compatible.
  599.      *
  600.      * @param left Left hand side matrix.
  601.      * @param right Right hand side matrix.
  602.      * @throws MathIllegalArgumentException if the matrices are not addition
  603.      * compatible.
  604.      */
  605.     public static void checkAdditionCompatible(final AnyMatrix left, final AnyMatrix right)
  606.         throws MathIllegalArgumentException {
  607.         if ((left.getRowDimension()    != right.getRowDimension()) ||
  608.             (left.getColumnDimension() != right.getColumnDimension())) {
  609.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH_2x2,
  610.                                                    left.getRowDimension(), left.getColumnDimension(),
  611.                                                    right.getRowDimension(), right.getColumnDimension());
  612.         }
  613.     }

  614.     /**
  615.      * Check if matrices are subtraction compatible
  616.      *
  617.      * @param left Left hand side matrix.
  618.      * @param right Right hand side matrix.
  619.      * @throws MathIllegalArgumentException if the matrices are not addition
  620.      * compatible.
  621.      */
  622.     public static void checkSubtractionCompatible(final AnyMatrix left, final AnyMatrix right)
  623.         throws MathIllegalArgumentException {
  624.         if ((left.getRowDimension()    != right.getRowDimension()) ||
  625.             (left.getColumnDimension() != right.getColumnDimension())) {
  626.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH_2x2,
  627.                                                    left.getRowDimension(), left.getColumnDimension(),
  628.                                                    right.getRowDimension(), right.getColumnDimension());
  629.         }
  630.     }

  631.     /**
  632.      * Check if matrices are multiplication compatible
  633.      *
  634.      * @param left Left hand side matrix.
  635.      * @param right Right hand side matrix.
  636.      * @throws MathIllegalArgumentException if matrices are not multiplication
  637.      * compatible.
  638.      */
  639.     public static void checkMultiplicationCompatible(final AnyMatrix left, final AnyMatrix right)
  640.         throws MathIllegalArgumentException {
  641.         if (left.getColumnDimension() != right.getRowDimension()) {
  642.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH,
  643.                                                    left.getColumnDimension(), right.getRowDimension());
  644.         }
  645.     }

  646.     /**
  647.      * Check if matrices have the same number of columns.
  648.      *
  649.      * @param left Left hand side matrix.
  650.      * @param right Right hand side matrix.
  651.      * @throws MathIllegalArgumentException if matrices don't have the same number of columns.
  652.      * @since 1.3
  653.      */
  654.     public static void checkSameColumnDimension(final AnyMatrix left, final AnyMatrix right)
  655.         throws MathIllegalArgumentException {
  656.         if (left.getColumnDimension() != right.getColumnDimension()) {
  657.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH,
  658.                                                    left.getColumnDimension(), right.getColumnDimension());
  659.         }
  660.     }

  661.     /**
  662.      * Check if matrices have the same number of rows.
  663.      *
  664.      * @param left Left hand side matrix.
  665.      * @param right Right hand side matrix.
  666.      * @throws MathIllegalArgumentException if matrices don't have the same number of rows.
  667.      * @since 1.3
  668.      */
  669.     public static void checkSameRowDimension(final AnyMatrix left, final AnyMatrix right)
  670.         throws MathIllegalArgumentException {
  671.         if (left.getRowDimension() != right.getRowDimension()) {
  672.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH,
  673.                                                    left.getRowDimension(), right.getRowDimension());
  674.         }
  675.     }

  676.     /**
  677.      * Convert a {@link FieldMatrix}/{@link Fraction} matrix to a {@link RealMatrix}.
  678.      * @param m Matrix to convert.
  679.      * @return the converted matrix.
  680.      */
  681.     public static Array2DRowRealMatrix fractionMatrixToRealMatrix(final FieldMatrix<Fraction> m) {
  682.         final FractionMatrixConverter converter = new FractionMatrixConverter();
  683.         m.walkInOptimizedOrder(converter);
  684.         return converter.getConvertedMatrix();
  685.     }

  686.     /** Converter for {@link FieldMatrix}/{@link Fraction}. */
  687.     private static class FractionMatrixConverter extends DefaultFieldMatrixPreservingVisitor<Fraction> {
  688.         /** Converted array. */
  689.         private double[][] data;
  690.         /** Simple constructor. */
  691.         FractionMatrixConverter() {
  692.             super(Fraction.ZERO);
  693.         }

  694.         /** {@inheritDoc} */
  695.         @Override
  696.         public void start(int rows, int columns,
  697.                           int startRow, int endRow, int startColumn, int endColumn) {
  698.             data = new double[rows][columns];
  699.         }

  700.         /** {@inheritDoc} */
  701.         @Override
  702.         public void visit(int row, int column, Fraction value) {
  703.             data[row][column] = value.doubleValue();
  704.         }

  705.         /**
  706.          * Get the converted matrix.
  707.          *
  708.          * @return the converted matrix.
  709.          */
  710.         Array2DRowRealMatrix getConvertedMatrix() {
  711.             return new Array2DRowRealMatrix(data, false);
  712.         }

  713.     }

  714.     /**
  715.      * Convert a {@link FieldMatrix}/{@link BigFraction} matrix to a {@link RealMatrix}.
  716.      *
  717.      * @param m Matrix to convert.
  718.      * @return the converted matrix.
  719.      */
  720.     public static Array2DRowRealMatrix bigFractionMatrixToRealMatrix(final FieldMatrix<BigFraction> m) {
  721.         final BigFractionMatrixConverter converter = new BigFractionMatrixConverter();
  722.         m.walkInOptimizedOrder(converter);
  723.         return converter.getConvertedMatrix();
  724.     }

  725.     /** Converter for {@link FieldMatrix}/{@link BigFraction}. */
  726.     private static class BigFractionMatrixConverter extends DefaultFieldMatrixPreservingVisitor<BigFraction> {
  727.         /** Converted array. */
  728.         private double[][] data;
  729.         /** Simple constructor. */
  730.         BigFractionMatrixConverter() {
  731.             super(BigFraction.ZERO);
  732.         }

  733.         /** {@inheritDoc} */
  734.         @Override
  735.         public void start(int rows, int columns,
  736.                           int startRow, int endRow, int startColumn, int endColumn) {
  737.             data = new double[rows][columns];
  738.         }

  739.         /** {@inheritDoc} */
  740.         @Override
  741.         public void visit(int row, int column, BigFraction value) {
  742.             data[row][column] = value.doubleValue();
  743.         }

  744.         /**
  745.          * Get the converted matrix.
  746.          *
  747.          * @return the converted matrix.
  748.          */
  749.         Array2DRowRealMatrix getConvertedMatrix() {
  750.             return new Array2DRowRealMatrix(data, false);
  751.         }
  752.     }

  753.     /**Solve  a  system of composed of a Lower Triangular Matrix
  754.      * {@link RealMatrix}.
  755.      * <p>
  756.      * This method is called to solve systems of equations which are
  757.      * of the lower triangular form. The matrix {@link RealMatrix}
  758.      * is assumed, though not checked, to be in lower triangular form.
  759.      * The vector {@link RealVector} is overwritten with the solution.
  760.      * The matrix is checked that it is square and its dimensions match
  761.      * the length of the vector.
  762.      * </p>
  763.      * @param rm RealMatrix which is lower triangular
  764.      * @param b  RealVector this is overwritten
  765.      * @throws MathIllegalArgumentException if the matrix and vector are not
  766.      * conformable
  767.      * @throws MathIllegalArgumentException if the matrix {@code rm} is not square
  768.      * @throws MathRuntimeException if the absolute value of one of the diagonal
  769.      * coefficient of {@code rm} is lower than {@link Precision#SAFE_MIN}
  770.      */
  771.     public static void solveLowerTriangularSystem(RealMatrix rm, RealVector b)
  772.         throws MathIllegalArgumentException, MathRuntimeException {
  773.         if ((rm == null) || (b == null) || ( rm.getRowDimension() != b.getDimension())) {
  774.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH,
  775.                                                    (rm == null) ? 0 : rm.getRowDimension(),
  776.                                                    (b  == null) ? 0 : b.getDimension());
  777.         }
  778.         if( rm.getColumnDimension() != rm.getRowDimension() ){
  779.             throw new MathIllegalArgumentException(LocalizedCoreFormats.NON_SQUARE_MATRIX,
  780.                                                    rm.getRowDimension(), rm.getColumnDimension());
  781.         }
  782.         int rows = rm.getRowDimension();
  783.         for( int i = 0 ; i < rows ; i++ ){
  784.             double diag = rm.getEntry(i, i);
  785.             if( FastMath.abs(diag) < Precision.SAFE_MIN ){
  786.                 throw new MathRuntimeException(LocalizedCoreFormats.ZERO_DENOMINATOR);
  787.             }
  788.             double bi = b.getEntry(i)/diag;
  789.             b.setEntry(i,  bi );
  790.             for( int j = i+1; j< rows; j++ ){
  791.                 b.setEntry(j, b.getEntry(j)-bi*rm.getEntry(j,i)  );
  792.             }
  793.         }
  794.     }

  795.     /** Solver a  system composed  of an Upper Triangular Matrix
  796.      * {@link RealMatrix}.
  797.      * <p>
  798.      * This method is called to solve systems of equations which are
  799.      * of the lower triangular form. The matrix {@link RealMatrix}
  800.      * is assumed, though not checked, to be in upper triangular form.
  801.      * The vector {@link RealVector} is overwritten with the solution.
  802.      * The matrix is checked that it is square and its dimensions match
  803.      * the length of the vector.
  804.      * </p>
  805.      * @param rm RealMatrix which is upper triangular
  806.      * @param b  RealVector this is overwritten
  807.      * @throws MathIllegalArgumentException if the matrix and vector are not
  808.      * conformable
  809.      * @throws MathIllegalArgumentException if the matrix {@code rm} is not
  810.      * square
  811.      * @throws MathRuntimeException if the absolute value of one of the diagonal
  812.      * coefficient of {@code rm} is lower than {@link Precision#SAFE_MIN}
  813.      */
  814.     public static void solveUpperTriangularSystem(RealMatrix rm, RealVector b)
  815.         throws MathIllegalArgumentException, MathRuntimeException {
  816.         if ((rm == null) || (b == null) || ( rm.getRowDimension() != b.getDimension())) {
  817.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH,
  818.                                                    (rm == null) ? 0 : rm.getRowDimension(),
  819.                                                    (b  == null) ? 0 : b.getDimension());
  820.         }
  821.         if( rm.getColumnDimension() != rm.getRowDimension() ){
  822.             throw new MathIllegalArgumentException(LocalizedCoreFormats.NON_SQUARE_MATRIX,
  823.                                                    rm.getRowDimension(), rm.getColumnDimension());
  824.         }
  825.         int rows = rm.getRowDimension();
  826.         for( int i = rows-1 ; i >-1 ; i-- ){
  827.             double diag = rm.getEntry(i, i);
  828.             if( FastMath.abs(diag) < Precision.SAFE_MIN ){
  829.                 throw new MathRuntimeException(LocalizedCoreFormats.ZERO_DENOMINATOR);
  830.             }
  831.             double bi = b.getEntry(i)/diag;
  832.             b.setEntry(i,  bi );
  833.             for( int j = i-1; j>-1; j-- ){
  834.                 b.setEntry(j, b.getEntry(j)-bi*rm.getEntry(j,i)  );
  835.             }
  836.         }
  837.     }

  838.     /**
  839.      * Computes the inverse of the given matrix by splitting it into
  840.      * 4 sub-matrices.
  841.      *
  842.      * @param m Matrix whose inverse must be computed.
  843.      * @param splitIndex Index that determines the "split" line and
  844.      * column.
  845.      * The element corresponding to this index will part of the
  846.      * upper-left sub-matrix.
  847.      * @return the inverse of {@code m}.
  848.      * @throws MathIllegalArgumentException if {@code m} is not square.
  849.      */
  850.     public static RealMatrix blockInverse(RealMatrix m,
  851.                                           int splitIndex) {
  852.         final int n = m.getRowDimension();
  853.         if (m.getColumnDimension() != n) {
  854.             throw new MathIllegalArgumentException(LocalizedCoreFormats.NON_SQUARE_MATRIX,
  855.                                                    m.getRowDimension(), m.getColumnDimension());
  856.         }

  857.         final int splitIndex1 = splitIndex + 1;

  858.         final RealMatrix a = m.getSubMatrix(0, splitIndex, 0, splitIndex);
  859.         final RealMatrix b = m.getSubMatrix(0, splitIndex, splitIndex1, n - 1);
  860.         final RealMatrix c = m.getSubMatrix(splitIndex1, n - 1, 0, splitIndex);
  861.         final RealMatrix d = m.getSubMatrix(splitIndex1, n - 1, splitIndex1, n - 1);

  862.         final SingularValueDecomposition aDec = new SingularValueDecomposition(a);
  863.         final DecompositionSolver aSolver = aDec.getSolver();
  864.         if (!aSolver.isNonSingular()) {
  865.             throw new MathIllegalArgumentException(LocalizedCoreFormats.SINGULAR_MATRIX);
  866.         }
  867.         final RealMatrix aInv = aSolver.getInverse();

  868.         final SingularValueDecomposition dDec = new SingularValueDecomposition(d);
  869.         final DecompositionSolver dSolver = dDec.getSolver();
  870.         if (!dSolver.isNonSingular()) {
  871.             throw new MathIllegalArgumentException(LocalizedCoreFormats.SINGULAR_MATRIX);
  872.         }
  873.         final RealMatrix dInv = dSolver.getInverse();

  874.         final RealMatrix tmp1 = a.subtract(b.multiply(dInv).multiply(c));
  875.         final SingularValueDecomposition tmp1Dec = new SingularValueDecomposition(tmp1);
  876.         final DecompositionSolver tmp1Solver = tmp1Dec.getSolver();
  877.         if (!tmp1Solver.isNonSingular()) {
  878.             throw new MathIllegalArgumentException(LocalizedCoreFormats.SINGULAR_MATRIX);
  879.         }
  880.         final RealMatrix result00 = tmp1Solver.getInverse();

  881.         final RealMatrix tmp2 = d.subtract(c.multiply(aInv).multiply(b));
  882.         final SingularValueDecomposition tmp2Dec = new SingularValueDecomposition(tmp2);
  883.         final DecompositionSolver tmp2Solver = tmp2Dec.getSolver();
  884.         if (!tmp2Solver.isNonSingular()) {
  885.             throw new MathIllegalArgumentException(LocalizedCoreFormats.SINGULAR_MATRIX);
  886.         }
  887.         final RealMatrix result11 = tmp2Solver.getInverse();

  888.         final RealMatrix result01 = aInv.multiply(b).multiply(result11).scalarMultiply(-1);
  889.         final RealMatrix result10 = dInv.multiply(c).multiply(result00).scalarMultiply(-1);

  890.         final RealMatrix result = new Array2DRowRealMatrix(n, n);
  891.         result.setSubMatrix(result00.getData(), 0, 0);
  892.         result.setSubMatrix(result01.getData(), 0, splitIndex1);
  893.         result.setSubMatrix(result10.getData(), splitIndex1, 0);
  894.         result.setSubMatrix(result11.getData(), splitIndex1, splitIndex1);

  895.         return result;
  896.     }

  897.     /**
  898.      * Computes the inverse of the given matrix.
  899.      * <p>
  900.      * By default, the inverse of the matrix is computed using the QR-decomposition,
  901.      * unless a more efficient method can be determined for the input matrix.
  902.      * <p>
  903.      * Note: this method will use a singularity threshold of 0,
  904.      * use {@link #inverse(RealMatrix, double)} if a different threshold is needed.
  905.      *
  906.      * @param matrix Matrix whose inverse shall be computed
  907.      * @return the inverse of {@code matrix}
  908.      * @throws NullArgumentException if {@code matrix} is {@code null}
  909.      * @throws MathIllegalArgumentException if m is singular
  910.      * @throws MathIllegalArgumentException if matrix is not square
  911.      */
  912.     public static RealMatrix inverse(RealMatrix matrix)
  913.             throws MathIllegalArgumentException, NullArgumentException {
  914.         return inverse(matrix, 0);
  915.     }

  916.     /**
  917.      * Computes the inverse of the given matrix.
  918.      * <p>
  919.      * By default, the inverse of the matrix is computed using the QR-decomposition,
  920.      * unless a more efficient method can be determined for the input matrix.
  921.      *
  922.      * @param matrix Matrix whose inverse shall be computed
  923.      * @param threshold Singularity threshold
  924.      * @return the inverse of {@code m}
  925.      * @throws NullArgumentException if {@code matrix} is {@code null}
  926.      * @throws MathIllegalArgumentException if matrix is singular
  927.      * @throws MathIllegalArgumentException if matrix is not square
  928.      */
  929.     public static RealMatrix inverse(RealMatrix matrix, double threshold)
  930.             throws MathIllegalArgumentException, NullArgumentException {

  931.         MathUtils.checkNotNull(matrix);

  932.         if (!matrix.isSquare()) {
  933.             throw new MathIllegalArgumentException(LocalizedCoreFormats.NON_SQUARE_MATRIX,
  934.                                                    matrix.getRowDimension(), matrix.getColumnDimension());
  935.         }

  936.         if (matrix instanceof DiagonalMatrix) {
  937.             return ((DiagonalMatrix) matrix).inverse(threshold);
  938.         } else {
  939.             QRDecomposition decomposition = new QRDecomposition(matrix, threshold);
  940.             return decomposition.getSolver().getInverse();
  941.         }
  942.     }

  943.     /**
  944.      * Computes the <a href="https://mathworld.wolfram.com/MatrixExponential.html">
  945.      * matrix exponential</a> of the given matrix.
  946.      *
  947.      * The algorithm implementation follows the Pade approximant method of
  948.      * <p>Higham, Nicholas J. “The Scaling and Squaring Method for the Matrix Exponential
  949.      * Revisited.” SIAM Journal on Matrix Analysis and Applications 26, no. 4 (January 2005): 1179–93.</p>
  950.      *
  951.      * @param rm RealMatrix whose inverse shall be computed
  952.      * @return The inverse of {@code rm}
  953.      * @throws MathIllegalArgumentException if matrix is not square
  954.      */
  955.     public static RealMatrix matrixExponential(final RealMatrix rm) {

  956.         // Check that the input matrix is square
  957.         if (!rm.isSquare()) {
  958.             throw new MathIllegalArgumentException(LocalizedCoreFormats.NON_SQUARE_MATRIX,
  959.                                                    rm.getRowDimension(), rm.getColumnDimension());
  960.         }

  961.         // Copy input matrix
  962.         int dim = rm.getRowDimension();
  963.         final RealMatrix identity = createRealIdentityMatrix(dim);
  964.         RealMatrix scaledMatrix = rm.copy();

  965.         // Select pade degree required
  966.         final double l1Norm = rm.getNorm1();
  967.         double[] padeCoefficients;
  968.         int squaringCount = 0;

  969.         if (l1Norm < 1.495585217958292e-2) {
  970.             padeCoefficients = PADE_COEFFICIENTS_3;
  971.         } else if (l1Norm < 2.539398330063230e-1) {
  972.             padeCoefficients = PADE_COEFFICIENTS_5;
  973.         } else if (l1Norm < 9.504178996162932e-1) {
  974.             padeCoefficients = PADE_COEFFICIENTS_7;
  975.         } else if (l1Norm < 2.097847961257068) {
  976.             padeCoefficients = PADE_COEFFICIENTS_9;
  977.         } else {
  978.             padeCoefficients = PADE_COEFFICIENTS_13;

  979.             // Calculate scaling factor
  980.             final double normScale = 5.371920351148152;
  981.             squaringCount = FastMath.max(0, FastMath.getExponent(l1Norm / normScale));

  982.             // Scale matrix by power of 2
  983.             final int finalSquaringCount = squaringCount;
  984.             scaledMatrix.mapToSelf(x -> FastMath.scalb(x, -finalSquaringCount));
  985.         }

  986.         // Calculate U and V using Horner
  987.         // See Golub, Gene H., and Charles F. van Loan. Matrix Computations. 4th ed.
  988.         // John Hopkins University Press, 2013.  pages 530/531
  989.         final RealMatrix scaledMatrix2 = scaledMatrix.multiply(scaledMatrix);
  990.         final int coeffLength = padeCoefficients.length;

  991.         // Calculate V
  992.         RealMatrix padeV = createRealMatrix(dim, dim);
  993.         for (int i = coeffLength - 1; i > 1; i -= 2) {
  994.             padeV = scaledMatrix2.multiply(padeV.add(identity.scalarMultiply(padeCoefficients[i])));
  995.         }
  996.         padeV = scaledMatrix.multiply(padeV.add(identity.scalarMultiply(padeCoefficients[1])));

  997.         // Calculate U
  998.         RealMatrix padeU = createRealMatrix(dim, dim);
  999.         for (int i = coeffLength - 2; i > 1; i -= 2) {
  1000.             padeU = scaledMatrix2.multiply(padeU.add(identity.scalarMultiply(padeCoefficients[i])));
  1001.         }
  1002.         padeU = padeU.add(identity.scalarMultiply(padeCoefficients[0]));

  1003.         // Calculate pade approximate by solving (U-V) F = (U+V) for F
  1004.         RealMatrix padeNumer = padeU.add(padeV);
  1005.         RealMatrix padeDenom = padeU.subtract(padeV);

  1006.         // Calculate the matrix ratio
  1007.         QRDecomposition decomposition = new QRDecomposition(padeDenom);
  1008.         RealMatrix result = decomposition.getSolver().solve(padeNumer);

  1009.         // Repeated squaring if matrix was scaled
  1010.         for (int i = 0; i < squaringCount; i++) {
  1011.             result = result.multiply(result);
  1012.         }

  1013.         return result;
  1014.     }

  1015.     /** Orthonormalize a list of vectors.
  1016.      * <p>
  1017.      * Orthonormalization is performed by using the Modified Gram-Schmidt process.
  1018.      * </p>
  1019.      * @param independent list of independent vectors
  1020.      * @param threshold projected vectors with a norm less than or equal to this threshold
  1021.      * are considered to have zero norm, hence the vectors they come from are not independent from
  1022.      * previous vectors
  1023.      * @param handler handler for dependent vectors
  1024.      * @return orthonormal basis having the same span as {@code independent}
  1025.      * @since 2.1
  1026.      */
  1027.     public static List<RealVector> orthonormalize(final List<RealVector> independent,
  1028.                                                   final double threshold, final DependentVectorsHandler handler) {

  1029.         // create separate list
  1030.         final List<RealVector> basis = new ArrayList<>(independent);

  1031.         // loop over basis vectors
  1032.         int index = 0;
  1033.         while (index < basis.size()) {

  1034.             // check dependency
  1035.             final RealVector vi = basis.get(index);
  1036.             final double norm = vi.getNorm();
  1037.             if (norm <= threshold) {
  1038.                 // the current vector is dependent from the previous ones
  1039.                 index = handler.manageDependent(index, basis);
  1040.             } else {

  1041.                 // normalize basis vector in place
  1042.                 vi.mapDivideToSelf(vi.getNorm());

  1043.                 // project remaining vectors in place
  1044.                 for (int j = index + 1; j < basis.size(); ++j) {
  1045.                     final RealVector vj  = basis.get(j);
  1046.                     final double     dot = vi.dotProduct(vj);
  1047.                     for (int k = 0; k < vj.getDimension(); ++k) {
  1048.                         vj.setEntry(k, vj.getEntry(k) - dot * vi.getEntry(k));
  1049.                     }
  1050.                 }

  1051.                 ++index;

  1052.             }

  1053.         }

  1054.         return basis;

  1055.     }

  1056.     /** Orthonormalize a list of vectors.
  1057.      * <p>
  1058.      * Orthonormalization is performed by using the Modified Gram-Schmidt process.
  1059.      * </p>
  1060.      * @param <T> type of the field elements
  1061.      * @param independent list of independent vectors
  1062.      * @param threshold projected vectors with a norm less than or equal to this threshold
  1063.      * are considered to have zero norm, hence the vectors they come from are not independent from
  1064.      * previous vectors
  1065.      * @param field type of the files elements
  1066.      * @param handler handler for dependent vectors
  1067.      * @return orthonormal basis having the same span as {@code independent}
  1068.      * @since 2.1
  1069.      */
  1070.     public static <T extends CalculusFieldElement<T>> List<FieldVector<T>> orthonormalize(final Field<T> field,
  1071.                                                                                           final List<FieldVector<T>> independent,
  1072.                                                                                           final T threshold,
  1073.                                                                                           final DependentVectorsHandler handler) {

  1074.         // create separate list
  1075.         final List<FieldVector<T>> basis = new ArrayList<>(independent);

  1076.         // loop over basis vectors
  1077.         int index = 0;
  1078.         while (index < basis.size()) {

  1079.             // check dependency
  1080.             final FieldVector<T> vi = basis.get(index);
  1081.             final T norm = vi.dotProduct(vi).sqrt();
  1082.             if (norm.subtract(threshold).getReal() <= 0) {
  1083.                 // the current vector is dependent from the previous ones
  1084.                 index = handler.manageDependent(field, index, basis);
  1085.             } else {

  1086.                 // normalize basis vector in place
  1087.                 vi.mapDivideToSelf(norm);

  1088.                 // project remaining vectors in place
  1089.                 for (int j = index + 1; j < basis.size(); ++j) {
  1090.                     final FieldVector<T> vj  = basis.get(j);
  1091.                     final T              dot = vi.dotProduct(vj);
  1092.                     for (int k = 0; k < vj.getDimension(); ++k) {
  1093.                         vj.setEntry(k, vj.getEntry(k).subtract(dot.multiply(vi.getEntry(k))));
  1094.                     }
  1095.                 }

  1096.                 ++index;

  1097.             }
  1098.         }

  1099.         return basis;

  1100.     }

  1101. }