AbstractFieldMatrix.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.List;

  24. import org.hipparchus.Field;
  25. import org.hipparchus.FieldElement;
  26. import org.hipparchus.exception.LocalizedCoreFormats;
  27. import org.hipparchus.exception.MathIllegalArgumentException;
  28. import org.hipparchus.exception.NullArgumentException;
  29. import org.hipparchus.util.MathArrays;

  30. /**
  31.  * Basic implementation of {@link FieldMatrix} methods regardless of the underlying storage.
  32.  * <p>All the methods implemented here use {@link #getEntry(int, int)} to access
  33.  * matrix elements. Derived class can provide faster implementations. </p>
  34.  *
  35.  * @param <T> Type of the field elements.
  36.  *
  37.  */
  38. public abstract class AbstractFieldMatrix<T extends FieldElement<T>>
  39.     implements FieldMatrix<T> {
  40.     /** Field to which the elements belong. */
  41.     private final Field<T> field;

  42.     /**
  43.      * Constructor for use with Serializable
  44.      */
  45.     protected AbstractFieldMatrix() {
  46.         field = null;
  47.     }

  48.     /**
  49.      * Creates a matrix with no data
  50.      * @param field field to which the elements belong
  51.      */
  52.     protected AbstractFieldMatrix(final Field<T> field) {
  53.         this.field = field;
  54.     }

  55.     /**
  56.      * Create a new {@link FieldMatrix} with the supplied row and column dimensions.
  57.      *
  58.      * @param field Field to which the elements belong.
  59.      * @param rowDimension Number of rows in the new matrix.
  60.      * @param columnDimension Number of columns in the new matrix.
  61.      * @throws MathIllegalArgumentException if row or column dimension is not
  62.      * positive.
  63.      */
  64.     protected AbstractFieldMatrix(final Field<T> field,
  65.                                   final int rowDimension,
  66.                                   final int columnDimension)
  67.         throws MathIllegalArgumentException {
  68.         if (rowDimension <= 0) {
  69.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSION,
  70.                                                    rowDimension);
  71.         }
  72.         if (columnDimension <= 0) {
  73.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSION,
  74.                                                    columnDimension);
  75.         }
  76.         this.field = field;
  77.     }

  78.     /**
  79.      * Get the elements type from an array.
  80.      *
  81.      * @param <T> Type of the field elements.
  82.      * @param d Data array.
  83.      * @return the field to which the array elements belong.
  84.      * @throws NullArgumentException if the array is {@code null}.
  85.      * @throws MathIllegalArgumentException if the array is empty.
  86.      */
  87.     protected static <T extends FieldElement<T>> Field<T> extractField(final T[][] d)
  88.         throws MathIllegalArgumentException, NullArgumentException {
  89.         if (d == null) {
  90.             throw new NullArgumentException();
  91.         }
  92.         if (d.length == 0) {
  93.             throw new MathIllegalArgumentException(LocalizedCoreFormats.AT_LEAST_ONE_ROW);
  94.         }
  95.         if (d[0].length == 0) {
  96.             throw new MathIllegalArgumentException(LocalizedCoreFormats.AT_LEAST_ONE_COLUMN);
  97.         }
  98.         return d[0][0].getField();
  99.     }

  100.     /**
  101.      * Get the elements type from an array.
  102.      *
  103.      * @param <T> Type of the field elements.
  104.      * @param d Data array.
  105.      * @return the field to which the array elements belong.
  106.      * @throws MathIllegalArgumentException if array is empty.
  107.      */
  108.     protected static <T extends FieldElement<T>> Field<T> extractField(final T[] d)
  109.         throws MathIllegalArgumentException {
  110.         if (d.length == 0) {
  111.             throw new MathIllegalArgumentException(LocalizedCoreFormats.AT_LEAST_ONE_ROW);
  112.         }
  113.         return d[0].getField();
  114.     }

  115.     /** {@inheritDoc} */
  116.     @Override
  117.     public Field<T> getField() {
  118.         return field;
  119.     }

  120.     /** {@inheritDoc} */
  121.     @Override
  122.     public abstract FieldMatrix<T> createMatrix(int rowDimension, int columnDimension)
  123.         throws MathIllegalArgumentException;

  124.     /** {@inheritDoc} */
  125.     @Override
  126.     public abstract FieldMatrix<T> copy();

  127.     /** {@inheritDoc} */
  128.     @Override
  129.     public FieldMatrix<T> add(FieldMatrix<T> m)
  130.         throws MathIllegalArgumentException {
  131.         // safety check
  132.         checkAdditionCompatible(m);

  133.         final int rowCount    = getRowDimension();
  134.         final int columnCount = getColumnDimension();
  135.         final FieldMatrix<T> out = createMatrix(rowCount, columnCount);
  136.         for (int row = 0; row < rowCount; ++row) {
  137.             for (int col = 0; col < columnCount; ++col) {
  138.                 out.setEntry(row, col, getEntry(row, col).add(m.getEntry(row, col)));
  139.             }
  140.         }

  141.         return out;
  142.     }

  143.     /** {@inheritDoc} */
  144.     @Override
  145.     public FieldMatrix<T> subtract(final FieldMatrix<T> m)
  146.         throws MathIllegalArgumentException {
  147.         // safety check
  148.         checkSubtractionCompatible(m);

  149.         final int rowCount    = getRowDimension();
  150.         final int columnCount = getColumnDimension();
  151.         final FieldMatrix<T> out = createMatrix(rowCount, columnCount);
  152.         for (int row = 0; row < rowCount; ++row) {
  153.             for (int col = 0; col < columnCount; ++col) {
  154.                 out.setEntry(row, col, getEntry(row, col).subtract(m.getEntry(row, col)));
  155.             }
  156.         }

  157.         return out;
  158.     }

  159.     /** {@inheritDoc} */
  160.     @Override
  161.     public FieldMatrix<T> scalarAdd(final T d) {

  162.         final int rowCount    = getRowDimension();
  163.         final int columnCount = getColumnDimension();
  164.         final FieldMatrix<T> out = createMatrix(rowCount, columnCount);
  165.         for (int row = 0; row < rowCount; ++row) {
  166.             for (int col = 0; col < columnCount; ++col) {
  167.                 out.setEntry(row, col, getEntry(row, col).add(d));
  168.             }
  169.         }

  170.         return out;
  171.     }

  172.     /** {@inheritDoc} */
  173.     @Override
  174.     public FieldMatrix<T> scalarMultiply(final T d) {
  175.         final int rowCount    = getRowDimension();
  176.         final int columnCount = getColumnDimension();
  177.         final FieldMatrix<T> out = createMatrix(rowCount, columnCount);
  178.         for (int row = 0; row < rowCount; ++row) {
  179.             for (int col = 0; col < columnCount; ++col) {
  180.                 out.setEntry(row, col, getEntry(row, col).multiply(d));
  181.             }
  182.         }

  183.         return out;
  184.     }

  185.     /** {@inheritDoc} */
  186.     @Override
  187.     public FieldMatrix<T> multiply(final FieldMatrix<T> m)
  188.         throws MathIllegalArgumentException {
  189.         // safety check
  190.         checkMultiplicationCompatible(m);

  191.         final int nRows = getRowDimension();
  192.         final int nCols = m.getColumnDimension();
  193.         final int nSum  = getColumnDimension();
  194.         final FieldMatrix<T> out = createMatrix(nRows, nCols);
  195.         for (int row = 0; row < nRows; ++row) {
  196.             for (int col = 0; col < nCols; ++col) {
  197.                 T sum = field.getZero();
  198.                 for (int i = 0; i < nSum; ++i) {
  199.                     sum = sum.add(getEntry(row, i).multiply(m.getEntry(i, col)));
  200.                 }
  201.                 out.setEntry(row, col, sum);
  202.             }
  203.         }

  204.         return out;
  205.     }

  206.     /** {@inheritDoc} */
  207.     @Override
  208.     public FieldMatrix<T> preMultiply(final FieldMatrix<T> m)
  209.         throws MathIllegalArgumentException {
  210.         return m.multiply(this);
  211.     }

  212.     /** {@inheritDoc} */
  213.     @Override
  214.     public FieldMatrix<T> power(final int p) throws MathIllegalArgumentException {
  215.         if (p < 0) {
  216.             throw new MathIllegalArgumentException(LocalizedCoreFormats.NUMBER_TOO_SMALL, p, 0);
  217.         }

  218.         if (!isSquare()) {
  219.             throw new MathIllegalArgumentException(LocalizedCoreFormats.NON_SQUARE_MATRIX,
  220.                                                    getRowDimension(), getColumnDimension());
  221.         }

  222.         if (p == 0) {
  223.             return MatrixUtils.createFieldIdentityMatrix(this.getField(), this.getRowDimension());
  224.         }

  225.         if (p == 1) {
  226.             return this.copy();
  227.         }

  228.         final int power = p - 1;

  229.         /*
  230.          * Only log_2(p) operations is used by doing as follows:
  231.          * 5^214 = 5^128 * 5^64 * 5^16 * 5^4 * 5^2
  232.          *
  233.          * In general, the same approach is used for A^p.
  234.          */

  235.         final char[]        binaryRepresentation = Integer.toBinaryString(power).toCharArray();
  236.         final List<Integer> nonZeroPositions     = new ArrayList<>();

  237.         for (int i = 0; i < binaryRepresentation.length; ++i) {
  238.             if (binaryRepresentation[i] == '1') {
  239.                 final int pos = binaryRepresentation.length - i - 1;
  240.                 nonZeroPositions.add(pos);
  241.             }
  242.         }

  243.         List<FieldMatrix<T>> results = new ArrayList<>(binaryRepresentation.length);

  244.         results.add(0, this.copy());

  245.         for (int i = 1; i < binaryRepresentation.length; ++i) {
  246.             final FieldMatrix<T> s = results.get(i - 1);
  247.             final FieldMatrix<T> r = s.multiply(s);
  248.             results.add(i, r);
  249.         }

  250.         FieldMatrix<T> result = this.copy();

  251.         for (Integer i : nonZeroPositions) {
  252.             result = result.multiply(results.get(i));
  253.         }

  254.         return result;
  255.     }

  256.     /** {@inheritDoc} */
  257.     @Override
  258.     public T[][] getData() {
  259.         final T[][] data = MathArrays.buildArray(field, getRowDimension(), getColumnDimension());

  260.         for (int i = 0; i < data.length; ++i) {
  261.             final T[] dataI = data[i];
  262.             for (int j = 0; j < dataI.length; ++j) {
  263.                 dataI[j] = getEntry(i, j);
  264.             }
  265.         }

  266.         return data;
  267.     }

  268.     /** {@inheritDoc} */
  269.     @Override
  270.     public FieldMatrix<T> getSubMatrix(final int startRow, final int endRow,
  271.                                        final int startColumn, final int endColumn)
  272.         throws MathIllegalArgumentException {
  273.         checkSubMatrixIndex(startRow, endRow, startColumn, endColumn);

  274.         final FieldMatrix<T> subMatrix =
  275.             createMatrix(endRow - startRow + 1, endColumn - startColumn + 1);
  276.         for (int i = startRow; i <= endRow; ++i) {
  277.             for (int j = startColumn; j <= endColumn; ++j) {
  278.                 subMatrix.setEntry(i - startRow, j - startColumn, getEntry(i, j));
  279.             }
  280.         }

  281.         return subMatrix;

  282.     }

  283.     /** {@inheritDoc} */
  284.     @Override
  285.     public FieldMatrix<T> getSubMatrix(final int[] selectedRows,
  286.                                        final int[] selectedColumns)
  287.     throws MathIllegalArgumentException, NullArgumentException {

  288.         // safety checks
  289.         checkSubMatrixIndex(selectedRows, selectedColumns);

  290.         // copy entries
  291.         final FieldMatrix<T> subMatrix =
  292.             createMatrix(selectedRows.length, selectedColumns.length);
  293.         subMatrix.walkInOptimizedOrder(new DefaultFieldMatrixChangingVisitor<T>(field.getZero()) {

  294.             /** {@inheritDoc} */
  295.             @Override
  296.             public T visit(final int row, final int column, final T value) {
  297.                 return getEntry(selectedRows[row], selectedColumns[column]);
  298.             }

  299.         });

  300.         return subMatrix;

  301.     }

  302.     /** {@inheritDoc} */
  303.     @Override
  304.     public void copySubMatrix(final int startRow, final int endRow,
  305.                               final int startColumn, final int endColumn,
  306.                               final T[][] destination)
  307.     throws MathIllegalArgumentException {
  308.         // safety checks
  309.         checkSubMatrixIndex(startRow, endRow, startColumn, endColumn);
  310.         final int rowsCount    = endRow + 1 - startRow;
  311.         final int columnsCount = endColumn + 1 - startColumn;
  312.         if ((destination.length < rowsCount) || (destination[0].length < columnsCount)) {
  313.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH_2x2,
  314.                                                    destination.length, destination[0].length,
  315.                                                    rowsCount, columnsCount);
  316.         }

  317.         // copy entries
  318.         walkInOptimizedOrder(new DefaultFieldMatrixPreservingVisitor<T>(field.getZero()) {

  319.             /** Initial row index. */
  320.             private int startRow;

  321.             /** Initial column index. */
  322.             private int startColumn;

  323.             /** {@inheritDoc} */
  324.             @Override
  325.             public void start(final int rows, final int columns,
  326.                               final int startRow, final int endRow,
  327.                               final int startColumn, final int endColumn) {
  328.                 this.startRow    = startRow;
  329.                 this.startColumn = startColumn;
  330.             }

  331.             /** {@inheritDoc} */
  332.             @Override
  333.             public void visit(final int row, final int column, final T value) {
  334.                 destination[row - startRow][column - startColumn] = value;
  335.             }

  336.         }, startRow, endRow, startColumn, endColumn);

  337.     }

  338.     /** {@inheritDoc} */
  339.     @Override
  340.     public void copySubMatrix(int[] selectedRows, int[] selectedColumns, T[][] destination)
  341.         throws MathIllegalArgumentException, NullArgumentException {
  342.         // safety checks
  343.         checkSubMatrixIndex(selectedRows, selectedColumns);
  344.         if ((destination.length < selectedRows.length) ||
  345.             (destination[0].length < selectedColumns.length)) {
  346.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH_2x2,
  347.                                                    destination.length, destination[0].length,
  348.                                                    selectedRows.length, selectedColumns.length);
  349.         }

  350.         // copy entries
  351.         for (int i = 0; i < selectedRows.length; i++) {
  352.             final T[] destinationI = destination[i];
  353.             for (int j = 0; j < selectedColumns.length; j++) {
  354.                 destinationI[j] = getEntry(selectedRows[i], selectedColumns[j]);
  355.             }
  356.         }

  357.     }

  358.     /** {@inheritDoc} */
  359.     @Override
  360.     public void setSubMatrix(final T[][] subMatrix, final int row,
  361.                              final int column)
  362.         throws MathIllegalArgumentException, NullArgumentException {
  363.         if (subMatrix == null) {
  364.             throw new NullArgumentException();
  365.         }
  366.         final int nRows = subMatrix.length;
  367.         if (nRows == 0) {
  368.             throw new MathIllegalArgumentException(LocalizedCoreFormats.AT_LEAST_ONE_ROW);
  369.         }

  370.         final int nCols = subMatrix[0].length;
  371.         if (nCols == 0) {
  372.             throw new MathIllegalArgumentException(LocalizedCoreFormats.AT_LEAST_ONE_COLUMN);
  373.         }

  374.         for (int r = 1; r < nRows; ++r) {
  375.             if (subMatrix[r].length != nCols) {
  376.                 throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH,
  377.                                                        nCols, subMatrix[r].length);
  378.             }
  379.         }

  380.         checkRowIndex(row);
  381.         checkColumnIndex(column);
  382.         checkRowIndex(nRows + row - 1);
  383.         checkColumnIndex(nCols + column - 1);

  384.         for (int i = 0; i < nRows; ++i) {
  385.             for (int j = 0; j < nCols; ++j) {
  386.                 setEntry(row + i, column + j, subMatrix[i][j]);
  387.             }
  388.         }
  389.     }

  390.     /** {@inheritDoc} */
  391.     @Override
  392.     public FieldMatrix<T> getRowMatrix(final int row) throws MathIllegalArgumentException {
  393.         checkRowIndex(row);
  394.         final int nCols = getColumnDimension();
  395.         final FieldMatrix<T> out = createMatrix(1, nCols);
  396.         for (int i = 0; i < nCols; ++i) {
  397.             out.setEntry(0, i, getEntry(row, i));
  398.         }

  399.         return out;

  400.     }

  401.     /** {@inheritDoc} */
  402.     @Override
  403.     public void setRowMatrix(final int row, final FieldMatrix<T> matrix)
  404.         throws MathIllegalArgumentException {
  405.         checkRowIndex(row);
  406.         final int nCols = getColumnDimension();
  407.         if ((matrix.getRowDimension() != 1) ||
  408.             (matrix.getColumnDimension() != nCols)) {
  409.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH_2x2,
  410.                                                    matrix.getRowDimension(), matrix.getColumnDimension(),
  411.                                                    1, nCols);
  412.         }
  413.         for (int i = 0; i < nCols; ++i) {
  414.             setEntry(row, i, matrix.getEntry(0, i));
  415.         }

  416.     }

  417.     /** {@inheritDoc} */
  418.     @Override
  419.     public FieldMatrix<T> getColumnMatrix(final int column)
  420.     throws MathIllegalArgumentException {

  421.         checkColumnIndex(column);
  422.         final int nRows = getRowDimension();
  423.         final FieldMatrix<T> out = createMatrix(nRows, 1);
  424.         for (int i = 0; i < nRows; ++i) {
  425.             out.setEntry(i, 0, getEntry(i, column));
  426.         }

  427.         return out;

  428.     }

  429.     /** {@inheritDoc} */
  430.     @Override
  431.     public void setColumnMatrix(final int column, final FieldMatrix<T> matrix)
  432.         throws MathIllegalArgumentException {
  433.         checkColumnIndex(column);
  434.         final int nRows = getRowDimension();
  435.         if ((matrix.getRowDimension() != nRows) ||
  436.             (matrix.getColumnDimension() != 1)) {
  437.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH_2x2,
  438.                                                    matrix.getRowDimension(), matrix.getColumnDimension(),
  439.                                                    nRows, 1);
  440.         }
  441.         for (int i = 0; i < nRows; ++i) {
  442.             setEntry(i, column, matrix.getEntry(i, 0));
  443.         }

  444.     }

  445.     /** {@inheritDoc} */
  446.     @Override
  447.     public FieldVector<T> getRowVector(final int row)
  448.         throws MathIllegalArgumentException {
  449.         return new ArrayFieldVector<>(field, getRow(row), false);
  450.     }

  451.     /** {@inheritDoc} */
  452.     @Override
  453.     public void setRowVector(final int row, final FieldVector<T> vector)
  454.         throws MathIllegalArgumentException {
  455.         checkRowIndex(row);
  456.         final int nCols = getColumnDimension();
  457.         if (vector.getDimension() != nCols) {
  458.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH_2x2,
  459.                                                    1, vector.getDimension(),
  460.                                                    1, nCols);
  461.         }
  462.         for (int i = 0; i < nCols; ++i) {
  463.             setEntry(row, i, vector.getEntry(i));
  464.         }

  465.     }

  466.     /** {@inheritDoc} */
  467.     @Override
  468.     public FieldVector<T> getColumnVector(final int column)
  469.         throws MathIllegalArgumentException {
  470.         return new ArrayFieldVector<>(field, getColumn(column), false);
  471.     }

  472.     /** {@inheritDoc} */
  473.     @Override
  474.     public void setColumnVector(final int column, final FieldVector<T> vector)
  475.         throws MathIllegalArgumentException {

  476.         checkColumnIndex(column);
  477.         final int nRows = getRowDimension();
  478.         if (vector.getDimension() != nRows) {
  479.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH_2x2,
  480.                                                    vector.getDimension(), 1,
  481.                                                    nRows, 1);
  482.         }
  483.         for (int i = 0; i < nRows; ++i) {
  484.             setEntry(i, column, vector.getEntry(i));
  485.         }

  486.     }

  487.     /** {@inheritDoc} */
  488.     @Override
  489.     public T[] getRow(final int row) throws MathIllegalArgumentException {
  490.         checkRowIndex(row);
  491.         final int nCols = getColumnDimension();
  492.         final T[] out = MathArrays.buildArray(field, nCols);
  493.         for (int i = 0; i < nCols; ++i) {
  494.             out[i] = getEntry(row, i);
  495.         }

  496.         return out;

  497.     }

  498.     /** {@inheritDoc} */
  499.     @Override
  500.     public void setRow(final int row, final T[] array)
  501.         throws MathIllegalArgumentException {
  502.         checkRowIndex(row);
  503.         final int nCols = getColumnDimension();
  504.         if (array.length != nCols) {
  505.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH_2x2,
  506.                                                    1, array.length, 1, nCols);
  507.         }
  508.         for (int i = 0; i < nCols; ++i) {
  509.             setEntry(row, i, array[i]);
  510.         }

  511.     }

  512.     /** {@inheritDoc} */
  513.     @Override
  514.     public T[] getColumn(final int column) throws MathIllegalArgumentException {
  515.         checkColumnIndex(column);
  516.         final int nRows = getRowDimension();
  517.         final T[] out = MathArrays.buildArray(field, nRows);
  518.         for (int i = 0; i < nRows; ++i) {
  519.             out[i] = getEntry(i, column);
  520.         }

  521.         return out;

  522.     }

  523.     /** {@inheritDoc} */
  524.     @Override
  525.     public void setColumn(final int column, final T[] array)
  526.         throws MathIllegalArgumentException {
  527.         checkColumnIndex(column);
  528.         final int nRows = getRowDimension();
  529.         if (array.length != nRows) {
  530.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH_2x2,
  531.                                                    array.length, 1, nRows, 1);
  532.         }
  533.         for (int i = 0; i < nRows; ++i) {
  534.             setEntry(i, column, array[i]);
  535.         }
  536.     }

  537.     /** {@inheritDoc} */
  538.     @Override
  539.     public abstract T getEntry(int row, int column) throws MathIllegalArgumentException;

  540.     /** {@inheritDoc} */
  541.     @Override
  542.     public abstract void setEntry(int row, int column, T value) throws MathIllegalArgumentException;

  543.     /** {@inheritDoc} */
  544.     @Override
  545.     public abstract void addToEntry(int row, int column, T increment) throws MathIllegalArgumentException;

  546.     /** {@inheritDoc} */
  547.     @Override
  548.     public abstract void multiplyEntry(int row, int column, T factor) throws MathIllegalArgumentException;

  549.     /** {@inheritDoc} */
  550.     @Override
  551.     public FieldMatrix<T> transpose() {
  552.         final int nRows = getRowDimension();
  553.         final int nCols = getColumnDimension();
  554.         final FieldMatrix<T> out = createMatrix(nCols, nRows);
  555.         walkInOptimizedOrder(new DefaultFieldMatrixPreservingVisitor<T>(field.getZero()) {
  556.             /** {@inheritDoc} */
  557.             @Override
  558.             public void visit(final int row, final int column, final T value) {
  559.                 out.setEntry(column, row, value);
  560.             }
  561.         });

  562.         return out;
  563.     }

  564.     /** {@inheritDoc} */
  565.     @Override
  566.     public boolean isSquare() {
  567.         return getColumnDimension() == getRowDimension();
  568.     }

  569.     /** {@inheritDoc} */
  570.     @Override
  571.     public abstract int getRowDimension();

  572.     /** {@inheritDoc} */
  573.     @Override
  574.     public abstract int getColumnDimension();

  575.     /** {@inheritDoc} */
  576.     @Override
  577.     public T getTrace() throws MathIllegalArgumentException {
  578.         final int nRows = getRowDimension();
  579.         final int nCols = getColumnDimension();
  580.         if (nRows != nCols) {
  581.             throw new MathIllegalArgumentException(LocalizedCoreFormats.NON_SQUARE_MATRIX,
  582.                                                    nRows, nCols);
  583.        }
  584.         T trace = field.getZero();
  585.         for (int i = 0; i < nRows; ++i) {
  586.             trace = trace.add(getEntry(i, i));
  587.         }
  588.         return trace;
  589.     }

  590.     /** {@inheritDoc} */
  591.     @Override
  592.     public T[] operate(final T[] v) throws MathIllegalArgumentException {

  593.         final int nRows = getRowDimension();
  594.         final int nCols = getColumnDimension();
  595.         if (v.length != nCols) {
  596.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH,
  597.                                                    v.length, nCols);
  598.         }

  599.         final T[] out = MathArrays.buildArray(field, nRows);
  600.         for (int row = 0; row < nRows; ++row) {
  601.             T sum = field.getZero();
  602.             for (int i = 0; i < nCols; ++i) {
  603.                 sum = sum.add(getEntry(row, i).multiply(v[i]));
  604.             }
  605.             out[row] = sum;
  606.         }

  607.         return out;
  608.     }

  609.     /** {@inheritDoc} */
  610.     @Override
  611.     public FieldVector<T> operate(final FieldVector<T> v)
  612.         throws MathIllegalArgumentException {
  613.         if (v instanceof ArrayFieldVector) {
  614.             return new ArrayFieldVector<>(field, operate(((ArrayFieldVector<T>) v).getDataRef()), false);
  615.         } else {
  616.             final int nRows = getRowDimension();
  617.             final int nCols = getColumnDimension();
  618.             if (v.getDimension() != nCols) {
  619.                 throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH,
  620.                                                        v.getDimension(), nCols);
  621.             }

  622.             final T[] out = MathArrays.buildArray(field, nRows);
  623.             for (int row = 0; row < nRows; ++row) {
  624.                 T sum = field.getZero();
  625.                 for (int i = 0; i < nCols; ++i) {
  626.                     sum = sum.add(getEntry(row, i).multiply(v.getEntry(i)));
  627.                 }
  628.                 out[row] = sum;
  629.             }

  630.             return new ArrayFieldVector<>(field, out, false);
  631.         }
  632.     }

  633.     /** {@inheritDoc} */
  634.     @Override
  635.     public T[] preMultiply(final T[] v) throws MathIllegalArgumentException {

  636.         final int nRows = getRowDimension();
  637.         final int nCols = getColumnDimension();
  638.         if (v.length != nRows) {
  639.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH,
  640.                                                    v.length, nRows);
  641.         }

  642.         final T[] out = MathArrays.buildArray(field, nCols);
  643.         for (int col = 0; col < nCols; ++col) {
  644.             T sum = field.getZero();
  645.             for (int i = 0; i < nRows; ++i) {
  646.                 sum = sum.add(getEntry(i, col).multiply(v[i]));
  647.             }
  648.             out[col] = sum;
  649.         }

  650.         return out;
  651.     }

  652.     /** {@inheritDoc} */
  653.     @Override
  654.     public FieldVector<T> preMultiply(final FieldVector<T> v)
  655.         throws MathIllegalArgumentException {
  656.         if (v instanceof ArrayFieldVector) {
  657.             return new ArrayFieldVector<>(field, preMultiply(((ArrayFieldVector<T>) v).getDataRef()), false);
  658.         } else {
  659.             final int nRows = getRowDimension();
  660.             final int nCols = getColumnDimension();
  661.             if (v.getDimension() != nRows) {
  662.                 throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH,
  663.                                                        v.getDimension(), nRows);
  664.             }

  665.             final T[] out = MathArrays.buildArray(field, nCols);
  666.             for (int col = 0; col < nCols; ++col) {
  667.                 T sum = field.getZero();
  668.                 for (int i = 0; i < nRows; ++i) {
  669.                     sum = sum.add(getEntry(i, col).multiply(v.getEntry(i)));
  670.                 }
  671.                 out[col] = sum;
  672.             }

  673.             return new ArrayFieldVector<>(field, out, false);
  674.         }
  675.     }

  676.     /** {@inheritDoc} */
  677.     @Override
  678.     public T walkInRowOrder(final FieldMatrixChangingVisitor<T> visitor) {
  679.         final int rows    = getRowDimension();
  680.         final int columns = getColumnDimension();
  681.         visitor.start(rows, columns, 0, rows - 1, 0, columns - 1);
  682.         for (int row = 0; row < rows; ++row) {
  683.             for (int column = 0; column < columns; ++column) {
  684.                 final T oldValue = getEntry(row, column);
  685.                 final T newValue = visitor.visit(row, column, oldValue);
  686.                 setEntry(row, column, newValue);
  687.             }
  688.         }
  689.         return visitor.end();
  690.     }

  691.     /** {@inheritDoc} */
  692.     @Override
  693.     public T walkInRowOrder(final FieldMatrixPreservingVisitor<T> visitor) {
  694.         final int rows    = getRowDimension();
  695.         final int columns = getColumnDimension();
  696.         visitor.start(rows, columns, 0, rows - 1, 0, columns - 1);
  697.         for (int row = 0; row < rows; ++row) {
  698.             for (int column = 0; column < columns; ++column) {
  699.                 visitor.visit(row, column, getEntry(row, column));
  700.             }
  701.         }
  702.         return visitor.end();
  703.     }

  704.     /** {@inheritDoc} */
  705.     @Override
  706.     public T walkInRowOrder(final FieldMatrixChangingVisitor<T> visitor,
  707.                             final int startRow, final int endRow,
  708.                             final int startColumn, final int endColumn)
  709.         throws MathIllegalArgumentException {
  710.         checkSubMatrixIndex(startRow, endRow, startColumn, endColumn);
  711.         visitor.start(getRowDimension(), getColumnDimension(),
  712.                       startRow, endRow, startColumn, endColumn);
  713.         for (int row = startRow; row <= endRow; ++row) {
  714.             for (int column = startColumn; column <= endColumn; ++column) {
  715.                 final T oldValue = getEntry(row, column);
  716.                 final T newValue = visitor.visit(row, column, oldValue);
  717.                 setEntry(row, column, newValue);
  718.             }
  719.         }
  720.         return visitor.end();
  721.     }

  722.     /** {@inheritDoc} */
  723.     @Override
  724.     public T walkInRowOrder(final FieldMatrixPreservingVisitor<T> visitor,
  725.                             final int startRow, final int endRow,
  726.                             final int startColumn, final int endColumn)
  727.         throws MathIllegalArgumentException {
  728.         checkSubMatrixIndex(startRow, endRow, startColumn, endColumn);
  729.         visitor.start(getRowDimension(), getColumnDimension(),
  730.                       startRow, endRow, startColumn, endColumn);
  731.         for (int row = startRow; row <= endRow; ++row) {
  732.             for (int column = startColumn; column <= endColumn; ++column) {
  733.                 visitor.visit(row, column, getEntry(row, column));
  734.             }
  735.         }
  736.         return visitor.end();
  737.     }

  738.     /** {@inheritDoc} */
  739.     @Override
  740.     public T walkInColumnOrder(final FieldMatrixChangingVisitor<T> visitor) {
  741.         final int rows    = getRowDimension();
  742.         final int columns = getColumnDimension();
  743.         visitor.start(rows, columns, 0, rows - 1, 0, columns - 1);
  744.         for (int column = 0; column < columns; ++column) {
  745.             for (int row = 0; row < rows; ++row) {
  746.                 final T oldValue = getEntry(row, column);
  747.                 final T newValue = visitor.visit(row, column, oldValue);
  748.                 setEntry(row, column, newValue);
  749.             }
  750.         }
  751.         return visitor.end();
  752.     }

  753.     /** {@inheritDoc} */
  754.     @Override
  755.     public T walkInColumnOrder(final FieldMatrixPreservingVisitor<T> visitor) {
  756.         final int rows    = getRowDimension();
  757.         final int columns = getColumnDimension();
  758.         visitor.start(rows, columns, 0, rows - 1, 0, columns - 1);
  759.         for (int column = 0; column < columns; ++column) {
  760.             for (int row = 0; row < rows; ++row) {
  761.                 visitor.visit(row, column, getEntry(row, column));
  762.             }
  763.         }
  764.         return visitor.end();
  765.     }

  766.     /** {@inheritDoc} */
  767.     @Override
  768.     public T walkInColumnOrder(final FieldMatrixChangingVisitor<T> visitor,
  769.                                final int startRow, final int endRow,
  770.                                final int startColumn, final int endColumn)
  771.     throws MathIllegalArgumentException {
  772.         checkSubMatrixIndex(startRow, endRow, startColumn, endColumn);
  773.         visitor.start(getRowDimension(), getColumnDimension(),
  774.                       startRow, endRow, startColumn, endColumn);
  775.         for (int column = startColumn; column <= endColumn; ++column) {
  776.             for (int row = startRow; row <= endRow; ++row) {
  777.                 final T oldValue = getEntry(row, column);
  778.                 final T newValue = visitor.visit(row, column, oldValue);
  779.                 setEntry(row, column, newValue);
  780.             }
  781.         }
  782.         return visitor.end();
  783.     }

  784.     /** {@inheritDoc} */
  785.     @Override
  786.     public T walkInColumnOrder(final FieldMatrixPreservingVisitor<T> visitor,
  787.                                final int startRow, final int endRow,
  788.                                final int startColumn, final int endColumn)
  789.     throws MathIllegalArgumentException {
  790.         checkSubMatrixIndex(startRow, endRow, startColumn, endColumn);
  791.         visitor.start(getRowDimension(), getColumnDimension(),
  792.                       startRow, endRow, startColumn, endColumn);
  793.         for (int column = startColumn; column <= endColumn; ++column) {
  794.             for (int row = startRow; row <= endRow; ++row) {
  795.                 visitor.visit(row, column, getEntry(row, column));
  796.             }
  797.         }
  798.         return visitor.end();
  799.     }

  800.     /** {@inheritDoc} */
  801.     @Override
  802.     public T walkInOptimizedOrder(final FieldMatrixChangingVisitor<T> visitor) {
  803.         return walkInRowOrder(visitor);
  804.     }

  805.     /** {@inheritDoc} */
  806.     @Override
  807.     public T walkInOptimizedOrder(final FieldMatrixPreservingVisitor<T> visitor) {
  808.         return walkInRowOrder(visitor);
  809.     }

  810.     /** {@inheritDoc} */
  811.     @Override
  812.     public T walkInOptimizedOrder(final FieldMatrixChangingVisitor<T> visitor,
  813.                                   final int startRow, final int endRow,
  814.                                   final int startColumn, final int endColumn)
  815.         throws MathIllegalArgumentException {
  816.         return walkInRowOrder(visitor, startRow, endRow, startColumn, endColumn);
  817.     }

  818.     /** {@inheritDoc} */
  819.     @Override
  820.     public T walkInOptimizedOrder(final FieldMatrixPreservingVisitor<T> visitor,
  821.                                   final int startRow, final int endRow,
  822.                                   final int startColumn, final int endColumn)
  823.         throws MathIllegalArgumentException {
  824.         return walkInRowOrder(visitor, startRow, endRow, startColumn, endColumn);
  825.     }

  826.     /**
  827.      * Get a string representation for this matrix.
  828.      * @return a string representation for this matrix
  829.      */
  830.     @Override
  831.     public String toString() {
  832.         final int nRows = getRowDimension();
  833.         final int nCols = getColumnDimension();
  834.         final StringBuffer res = new StringBuffer();
  835.         String fullClassName = getClass().getName();
  836.         String shortClassName = fullClassName.substring(fullClassName.lastIndexOf('.') + 1);
  837.         res.append(shortClassName).append('{');

  838.         for (int i = 0; i < nRows; ++i) {
  839.             if (i > 0) {
  840.                 res.append(',');
  841.             }
  842.             res.append('{');
  843.             for (int j = 0; j < nCols; ++j) {
  844.                 if (j > 0) {
  845.                     res.append(',');
  846.                 }
  847.                 res.append(getEntry(i, j));
  848.             }
  849.             res.append('}');
  850.         }

  851.         res.append('}');
  852.         return res.toString();
  853.     }

  854.     /**
  855.      * Returns true iff <code>object</code> is a
  856.      * <code>FieldMatrix</code> instance with the same dimensions as this
  857.      * and all corresponding matrix entries are equal.
  858.      *
  859.      * @param object the object to test equality against.
  860.      * @return true if object equals this
  861.      */
  862.     @Override
  863.     public boolean equals(final Object object) {
  864.         if (object == this) {
  865.             return true;
  866.         }
  867.         if (!(object instanceof FieldMatrix<?>)) {
  868.             return false;
  869.         }
  870.         FieldMatrix<?> m = (FieldMatrix<?>) object;
  871.         final int nRows = getRowDimension();
  872.         final int nCols = getColumnDimension();
  873.         if (m.getColumnDimension() != nCols || m.getRowDimension() != nRows) {
  874.             return false;
  875.         }
  876.         for (int row = 0; row < nRows; ++row) {
  877.             for (int col = 0; col < nCols; ++col) {
  878.                 if (!getEntry(row, col).equals(m.getEntry(row, col))) {
  879.                     return false;
  880.                 }
  881.             }
  882.         }
  883.         return true;
  884.     }

  885.     /**
  886.      * Computes a hashcode for the matrix.
  887.      *
  888.      * @return hashcode for matrix
  889.      */
  890.     @Override
  891.     public int hashCode() {
  892.         int ret = 322562;
  893.         final int nRows = getRowDimension();
  894.         final int nCols = getColumnDimension();
  895.         ret = ret * 31 + nRows;
  896.         ret = ret * 31 + nCols;
  897.         for (int row = 0; row < nRows; ++row) {
  898.             for (int col = 0; col < nCols; ++col) {
  899.                ret = ret * 31 + (11 * (row+1) + 17 * (col+1)) * getEntry(row, col).hashCode();
  900.            }
  901.         }
  902.         return ret;
  903.     }

  904.     /**
  905.      * Check if a row index is valid.
  906.      *
  907.      * @param row Row index to check.
  908.      * @throws MathIllegalArgumentException if {@code index} is not valid.
  909.      */
  910.     protected void checkRowIndex(final int row) throws MathIllegalArgumentException {
  911.         if (row < 0 || row >= getRowDimension()) {
  912.             throw new MathIllegalArgumentException(LocalizedCoreFormats.ROW_INDEX,
  913.                                           row, 0, getRowDimension() - 1);
  914.         }
  915.     }

  916.     /**
  917.      * Check if a column index is valid.
  918.      *
  919.      * @param column Column index to check.
  920.      * @throws MathIllegalArgumentException if {@code index} is not valid.
  921.      */
  922.     protected void checkColumnIndex(final int column)
  923.         throws MathIllegalArgumentException {
  924.         if (column < 0 || column >= getColumnDimension()) {
  925.             throw new MathIllegalArgumentException(LocalizedCoreFormats.COLUMN_INDEX,
  926.                                           column, 0, getColumnDimension() - 1);
  927.         }
  928.     }

  929.     /**
  930.      * Check if submatrix ranges indices are valid.
  931.      * Rows and columns are indicated counting from 0 to n-1.
  932.      *
  933.      * @param startRow Initial row index.
  934.      * @param endRow Final row index.
  935.      * @param startColumn Initial column index.
  936.      * @param endColumn Final column index.
  937.      * @throws MathIllegalArgumentException if the indices are not valid.
  938.      * @throws MathIllegalArgumentException if {@code endRow < startRow} or
  939.      * {@code endColumn < startColumn}.
  940.      */
  941.     protected void checkSubMatrixIndex(final int startRow, final int endRow,
  942.                                        final int startColumn, final int endColumn)
  943.         throws MathIllegalArgumentException {
  944.         checkRowIndex(startRow);
  945.         checkRowIndex(endRow);
  946.         if (endRow < startRow) {
  947.             throw new MathIllegalArgumentException(LocalizedCoreFormats.INITIAL_ROW_AFTER_FINAL_ROW,
  948.                                                 endRow, startRow, true);
  949.         }

  950.         checkColumnIndex(startColumn);
  951.         checkColumnIndex(endColumn);
  952.         if (endColumn < startColumn) {
  953.             throw new MathIllegalArgumentException(LocalizedCoreFormats.INITIAL_COLUMN_AFTER_FINAL_COLUMN,
  954.                                                 endColumn, startColumn, true);
  955.         }
  956.     }

  957.     /**
  958.      * Check if submatrix ranges indices are valid.
  959.      * Rows and columns are indicated counting from 0 to n-1.
  960.      *
  961.      * @param selectedRows Array of row indices.
  962.      * @param selectedColumns Array of column indices.
  963.      * @throws NullArgumentException if the arrays are {@code null}.
  964.      * @throws MathIllegalArgumentException if the arrays have zero length.
  965.      * @throws MathIllegalArgumentException if row or column selections are not valid.
  966.      */
  967.     protected void checkSubMatrixIndex(final int[] selectedRows, final int[] selectedColumns)
  968.         throws MathIllegalArgumentException, NullArgumentException {
  969.         if (selectedRows == null ||
  970.             selectedColumns == null) {
  971.             throw new NullArgumentException();
  972.         }
  973.         if (selectedRows.length == 0 ||
  974.             selectedColumns.length == 0) {
  975.             throw new MathIllegalArgumentException(LocalizedCoreFormats.NO_DATA);
  976.         }

  977.         for (final int row : selectedRows) {
  978.             checkRowIndex(row);
  979.         }
  980.         for (final int column : selectedColumns) {
  981.             checkColumnIndex(column);
  982.         }
  983.     }

  984.     /**
  985.      * Check if a matrix is addition compatible with the instance.
  986.      *
  987.      * @param m Matrix to check.
  988.      * @throws MathIllegalArgumentException if the matrix is not
  989.      * addition-compatible with instance.
  990.      */
  991.     protected void checkAdditionCompatible(final FieldMatrix<T> m)
  992.         throws MathIllegalArgumentException {
  993.         if ((getRowDimension() != m.getRowDimension()) ||
  994.             (getColumnDimension() != m.getColumnDimension())) {
  995.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH_2x2,
  996.                                                    m.getRowDimension(), m.getColumnDimension(),
  997.                                                    getRowDimension(), getColumnDimension());
  998.         }
  999.     }

  1000.     /**
  1001.      * Check if a matrix is subtraction compatible with the instance.
  1002.      *
  1003.      * @param m Matrix to check.
  1004.      * @throws MathIllegalArgumentException if the matrix is not
  1005.      * subtraction-compatible with instance.
  1006.      */
  1007.     protected void checkSubtractionCompatible(final FieldMatrix<T> m)
  1008.         throws MathIllegalArgumentException {
  1009.         if ((getRowDimension() != m.getRowDimension()) ||
  1010.             (getColumnDimension() != m.getColumnDimension())) {
  1011.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH_2x2,
  1012.                                                    m.getRowDimension(), m.getColumnDimension(),
  1013.                                                    getRowDimension(), getColumnDimension());
  1014.         }
  1015.     }

  1016.     /**
  1017.      * Check if a matrix is multiplication compatible with the instance.
  1018.      *
  1019.      * @param m Matrix to check.
  1020.      * @throws MathIllegalArgumentException if the matrix is not
  1021.      * multiplication-compatible with instance.
  1022.      */
  1023.     protected void checkMultiplicationCompatible(final FieldMatrix<T> m)
  1024.         throws MathIllegalArgumentException {
  1025.         if (getColumnDimension() != m.getRowDimension()) {
  1026.             throw new MathIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH,
  1027.                                                    getColumnDimension(), m.getRowDimension());
  1028.         }
  1029.     }
  1030. }