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

org.ojalgo.structure.Access2D Maven / Gradle / Ivy

Go to download

oj! Algorithms - ojAlgo - is Open Source Java code that has to do with mathematics, linear algebra and optimisation.

There is a newer version: 55.0.1
Show newest version
/*
 * Copyright 1997-2024 Optimatika
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 */
package org.ojalgo.structure;

import java.util.Iterator;
import java.util.Spliterator;
import java.util.function.Consumer;
import java.util.stream.Stream;
import java.util.stream.StreamSupport;

import org.ojalgo.ProgrammingError;
import org.ojalgo.function.VoidFunction;
import org.ojalgo.function.aggregator.Aggregator;
import org.ojalgo.matrix.store.RawStore;
import org.ojalgo.type.NumberDefinition;
import org.ojalgo.type.context.NumberContext;

/**
 * 2-dimensional accessor methods
 *
 * @see Access1D
 * @author apete
 */
public interface Access2D> extends Structure2D, Access1D {

    public interface Aggregatable> extends Structure2D, Access1D.Aggregatable {

        default N aggregateColumn(final long col, final Aggregator aggregator) {
            return this.aggregateColumn(0L, col, aggregator);
        }

        N aggregateColumn(long row, long col, Aggregator aggregator);

        default N aggregateDiagonal(final Aggregator aggregator) {
            return this.aggregateDiagonal(0L, 0L, aggregator);
        }

        N aggregateDiagonal(long row, long col, Aggregator aggregator);

        default N aggregateRow(final long row, final Aggregator aggregator) {
            return this.aggregateRow(row, 0L, aggregator);
        }

        N aggregateRow(long row, long col, Aggregator aggregator);

        default void reduceColumns(final Aggregator aggregator, final Mutate1D receiver) {
            for (long j = 0L, limit = Math.min(this.countColumns(), receiver.count()); j < limit; j++) {
                receiver.set(j, this.aggregateColumn(j, aggregator));
            }
        }

        default void reduceRows(final Aggregator aggregator, final Mutate1D receiver) {
            for (long i = 0L, limit = Math.min(this.countRows(), receiver.count()); i < limit; i++) {
                receiver.set(i, this.aggregateRow(i, aggregator));
            }
        }

    }

    public interface Collectable, R extends Mutate2D> extends Structure2D {

        default  I collect(final Factory2D factory) {

            I retVal = factory.make(this.countRows(), this.countColumns());

            this.supplyTo(retVal);

            return retVal;
        }

        void supplyTo(R receiver);

    }

    public static class ColumnView> implements Access1D, Iterable>, Iterator>,
            Spliterator>, Comparable>, Access1D.Collectable {

        static final int CHARACTERISTICS = Spliterator.CONCURRENT | Spliterator.DISTINCT | Spliterator.IMMUTABLE | Spliterator.NONNULL | Spliterator.ORDERED
                | Spliterator.SIZED | Spliterator.SORTED | Spliterator.SUBSIZED;

        private long myColumn = -1L;
        private final Access2D myDelegate2D;
        private final long myLastColumn;

        private ColumnView(final Access2D access, final long column, final long lastColumn) {

            super();

            myDelegate2D = access;
            myLastColumn = lastColumn;

            myColumn = column;
        }

        protected ColumnView(final Access2D access) {
            this(access, -1L, access.countColumns() - 1L);
        }

        ColumnView(final Access2D access, final long column) {
            this(access, column, access.countColumns() - 1L);
        }

        @Override
        public int characteristics() {
            return CHARACTERISTICS;
        }

        public long column() {
            return myColumn;
        }

        @Override
        public int compareTo(final ColumnView other) {
            return Long.compare(myColumn, other.column());
        }

        @Override
        public long count() {
            return myDelegate2D.countRows();
        }

        @Override
        public double doubleValue(final int index) {
            return myDelegate2D.doubleValue(index, myColumn);
        }

        @Override
        public long estimateSize() {
            return myLastColumn - myColumn;
        }

        @Override
        public void forEachRemaining(final Consumer> action) {
            Iterator.super.forEachRemaining(action);
        }

        @Override
        public N get(final long index) {
            return myDelegate2D.get(index, myColumn);
        }

        public void goToColumn(final long column) {
            myColumn = column;
        }

        @Override
        public boolean hasNext() {
            return myColumn < myLastColumn;
        }

        public boolean hasPrevious() {
            return myColumn > 0L;
        }

        @Override
        public ColumnView iterator() {
            return new ColumnView<>(myDelegate2D);
        }

        @Override
        public ColumnView next() {
            myColumn++;
            return this;
        }

        public ColumnView previous() {
            myColumn--;
            return this;
        }

        @Override
        public void remove() {
            ProgrammingError.throwForUnsupportedOptionalOperation();
        }

        @Override
        public int size() {
            return myDelegate2D.getRowDim();
        }

        public Stream> stream() {
            return StreamSupport.stream(this, false);
        }

        @Override
        public void supplyTo(final Mutate1D receiver) {
            for (long i = 0L, limit = Math.min(myDelegate2D.countRows(), receiver.count()); i < limit; i++) {
                receiver.set(i, myDelegate2D.get(i, myColumn));
            }
        }

        @Override
        public String toString() {
            return Access1D.toString(this);
        }

        @Override
        public boolean tryAdvance(final Consumer> action) {
            if (this.hasNext()) {
                action.accept(this.next());
                return true;
            } else {
                return false;
            }
        }

        @Override
        public Spliterator> trySplit() {

            final long remaining = myLastColumn - myColumn;

            if (remaining > 1L) {

                final long split = myColumn + remaining / 2L;

                final ColumnView retVal = new ColumnView<>(myDelegate2D, myColumn, split);

                myColumn = split;

                return retVal;

            } else {

                return null;
            }
        }

    }

    public static class ElementView> implements ElementView2D> {

        private final ElementView1D myDelegate1D;
        private final long myStructure;

        public ElementView(final ElementView1D delegate, final long structure) {

            super();

            myDelegate1D = delegate;
            myStructure = structure;
        }

        @Override
        public long column() {
            return Structure2D.column(myDelegate1D.index(), myStructure);
        }

        @Override
        public double doubleValue() {
            return myDelegate1D.doubleValue();
        }

        @Override
        public long estimateSize() {
            return myDelegate1D.estimateSize();
        }

        @Override
        public N get() {
            return myDelegate1D.get();
        }

        @Override
        public boolean hasNext() {
            return myDelegate1D.hasNext();
        }

        @Override
        public boolean hasPrevious() {
            return myDelegate1D.hasPrevious();
        }

        @Override
        public long index() {
            return myDelegate1D.index();
        }

        @Override
        public ElementView iterator() {
            return new ElementView<>(myDelegate1D.iterator(), myStructure);
        }

        @Override
        public ElementView next() {
            myDelegate1D.next();
            return this;
        }

        @Override
        public long nextIndex() {
            return myDelegate1D.nextIndex();
        }

        @Override
        public ElementView previous() {
            myDelegate1D.previous();
            return this;
        }

        @Override
        public long previousIndex() {
            return myDelegate1D.previousIndex();
        }

        @Override
        public long row() {
            return Structure2D.row(myDelegate1D.index(), myStructure);
        }

        @Override
        public String toString() {
            return myDelegate1D.toString();
        }

        @Override
        public ElementView trySplit() {

            ElementView1D delegateSpliterator = myDelegate1D.trySplit();

            if (delegateSpliterator != null) {
                return new ElementView<>(delegateSpliterator, myStructure);
            }
            return null;
        }

    }

    public static class RowView> implements Access1D, Iterable>, Iterator>, Spliterator>,
            Comparable>, Access1D.Collectable {

        static final int CHARACTERISTICS = Spliterator.CONCURRENT | Spliterator.DISTINCT | Spliterator.IMMUTABLE | Spliterator.NONNULL | Spliterator.ORDERED
                | Spliterator.SIZED | Spliterator.SORTED | Spliterator.SUBSIZED;

        private final Access2D myDelegate2D;
        private final long myLastRow;
        private long myRow = -1L;

        private RowView(final Access2D access, final long row, final long lastRow) {

            super();

            myDelegate2D = access;
            myLastRow = lastRow;

            myRow = row;
        }

        protected RowView(final Access2D access) {
            this(access, -1L, access.countRows() - 1L);
        }

        RowView(final Access2D access, final long row) {
            this(access, row, access.countRows() - 1L);
        }

        @Override
        public int characteristics() {
            return CHARACTERISTICS;
        }

        @Override
        public int compareTo(final RowView other) {
            return Long.compare(myRow, other.row());
        }

        @Override
        public long count() {
            return myDelegate2D.countColumns();
        }

        @Override
        public double doubleValue(final int index) {
            return myDelegate2D.doubleValue(myRow, index);
        }

        @Override
        public long estimateSize() {
            return myLastRow - myRow;
        }

        @Override
        public void forEachRemaining(final Consumer> action) {
            Iterator.super.forEachRemaining(action);
        }

        @Override
        public N get(final long index) {
            return myDelegate2D.get(myRow, index);
        }

        public void goToRow(final long row) {
            myRow = row;
        }

        @Override
        public boolean hasNext() {
            return myRow < myLastRow;
        }

        public boolean hasPrevious() {
            return myRow > 0L;
        }

        @Override
        public RowView iterator() {
            return new RowView<>(myDelegate2D);
        }

        @Override
        public RowView next() {
            myRow++;
            return this;
        }

        public RowView previous() {
            myRow--;
            return this;
        }

        @Override
        public void remove() {
            ProgrammingError.throwForUnsupportedOptionalOperation();
        }

        public long row() {
            return myRow;
        }

        @Override
        public int size() {
            return myDelegate2D.getColDim();
        }

        public Stream> stream() {
            return StreamSupport.stream(this, false);
        }

        @Override
        public void supplyTo(final Mutate1D receiver) {
            for (long j = 0L, limit = Math.min(myDelegate2D.countColumns(), receiver.count()); j < limit; j++) {
                receiver.set(j, myDelegate2D.get(myRow, j));
            }
        }

        @Override
        public String toString() {
            return Access1D.toString(this);
        }

        @Override
        public boolean tryAdvance(final Consumer> action) {
            if (this.hasNext()) {
                action.accept(this.next());
                return true;
            } else {
                return false;
            }
        }

        @Override
        public Spliterator> trySplit() {

            final long remaining = myLastRow - myRow;

            if (remaining > 1L) {

                final long split = myRow + remaining / 2L;

                final RowView retVal = new RowView<>(myDelegate2D, myRow, split);

                myRow = split;

                return retVal;

            } else {

                return null;
            }
        }

    }

    public static final class SelectionView> implements Access2D, Collectable {

        private final long[] myColumns;
        private final Access2D myFullData;
        private final long[] myRows;

        SelectionView(final Access2D fullData, final long[] rows, final long[] columns) {

            super();

            myFullData = fullData;
            myRows = Structure1D.replaceNullOrEmptyWithFull(rows, fullData.getRowDim());
            myColumns = Structure1D.replaceNullOrEmptyWithFull(columns, fullData.getColDim());
        }

        @Override
        public long countColumns() {
            return myColumns.length;
        }

        @Override
        public long countRows() {
            return myRows.length;
        }

        @Override
        public double doubleValue(final int row, final int col) {
            return myFullData.doubleValue(myRows[row], myColumns[col]);
        }

        @Override
        public double doubleValue(final long row, final long col) {
            return myFullData.doubleValue(myRows[Math.toIntExact(row)], myColumns[Math.toIntExact(col)]);
        }

        @Override
        public N get(final long row, final long col) {
            return myFullData.get(myRows[Math.toIntExact(row)], myColumns[Math.toIntExact(col)]);
        }

        @Override
        public int getColDim() {
            return myColumns.length;
        }

        @Override
        public int getRowDim() {
            return myRows.length;
        }

        @Override
        public void supplyTo(final Mutate2D receiver) {
            for (int j = 0; j < myColumns.length; j++) {
                for (int i = 0; i < myRows.length; i++) {
                    receiver.set(i, j, myFullData.get(myRows[i], myColumns[j]));
                }
            }
        }

        @Override
        public String toString() {
            return Access2D.toString(this);
        }

    }

    public interface Sliceable> extends Structure2D, Access1D.Sliceable {

        default Access1D sliceColumn(final long col) {
            return this.sliceColumn(0L, col);
            // return new ColumnView(this, col);
        }

        Access1D sliceColumn(long row, long col);

        default Access1D sliceDiagonal() {
            return this.sliceDiagonal(0L, 0L);
        }

        Access1D sliceDiagonal(long row, long col);

        default Access1D sliceRow(final long row) {
            return this.sliceRow(row, 0L);
        }

        Access1D sliceRow(long row, long col);

    }

    public interface Visitable> extends Structure2D, Access1D.Visitable {

        default void visitColumn(final long row, final long col, final VoidFunction visitor) {
            for (long i = row, limit = this.countRows(); i < limit; i++) {
                this.visitOne(i, col, visitor);
            }
        }

        default void visitColumn(final long col, final VoidFunction visitor) {
            this.visitColumn(0L, col, visitor);
        }

        default void visitDiagonal(final long row, final long col, final VoidFunction visitor) {
            for (long ij = 0L, limit = Math.min(this.countRows() - row, this.countColumns() - col); ij < limit; ij++) {
                this.visitOne(row + ij, col + ij, visitor);
            }
        }

        default void visitDiagonal(final VoidFunction visitor) {
            this.visitDiagonal(0L, 0L, visitor);
        }

        void visitOne(long row, long col, VoidFunction visitor);

        @Override
        default void visitOne(final long index, final VoidFunction visitor) {
            long tmpStructure = this.countRows();
            this.visitOne(Structure2D.row(index, tmpStructure), Structure2D.column(index, tmpStructure), visitor);
        }

        default void visitRow(final long row, final long col, final VoidFunction visitor) {
            for (long j = col, limit = this.countColumns(); j < limit; j++) {
                this.visitOne(row, j, visitor);
            }
        }

        default void visitRow(final long row, final VoidFunction visitor) {
            this.visitRow(row, 0L, visitor);
        }

    }

    /**
     * @deprecated v54 Use {@link Primitive2D#wrap(Structure2D)} instead
     */
    @Deprecated
    static Access2D asPrimitive2D(final Access2D access) {
        return Primitive2D.wrap(access);
    }

    static boolean equals(final Access2D accessA, final Access2D accessB, final NumberContext accuracy) {
        return accessA.countRows() == accessB.countRows() && accessA.countColumns() == accessB.countColumns() && Access1D.equals(accessA, accessB, accuracy);
    }

    static > Access2D.Collectable newPrimitiveColumnCollectable(final Access1D anything1D) {
        return new Access2D.Collectable<>() {

            public long countColumns() {
                return 1L;
            }

            public long countRows() {
                return anything1D.count();
            }

            public int getColDim() {
                return 1;
            }

            public int getRowDim() {
                return anything1D.size();
            }

            public void supplyTo(final R receiver) {
                receiver.reset();
                anything1D.nonzeros().forEach(nz -> receiver.set(nz.index(), 0L, nz.doubleValue()));
            }

        };
    }

    static > Access2D.Collectable newPrimitiveRowCollectable(final Access1D anything1D) {
        return new Access2D.Collectable<>() {

            public long countColumns() {
                return anything1D.count();
            }

            public long countRows() {
                return 1L;
            }

            public int getColDim() {
                return anything1D.size();
            }

            public int getRowDim() {
                return 1;
            }

            public void supplyTo(final R receiver) {
                receiver.reset();
                anything1D.nonzeros().forEach(nz -> receiver.set(0L, nz.index(), nz.doubleValue()));
            }

        };
    }

    static String toString(final Access2D matrix) {

        StringBuilder builder = new StringBuilder();

        int numbRows = Math.toIntExact(matrix.countRows());
        int numbCols = Math.toIntExact(matrix.countColumns());

        builder.append(matrix.getClass().getName());
        builder.append(' ').append('<').append(' ').append(numbRows).append(' ').append('x').append(' ').append(numbCols).append(' ').append('>');

        if (numbRows > 0 && numbCols > 0 && numbRows <= 50 && numbCols <= 50 && numbRows * numbCols <= 200) {

            // First element
            builder.append("\n{ { ").append(matrix.get(0, 0));

            // Rest of the first row
            for (int j = 1; j < numbCols; j++) {
                builder.append(",\t").append(matrix.get(0, j));
            }

            // For each of the remaining rows
            for (int i = 1; i < numbRows; i++) {

                // First column
                builder.append(" },\n{ ").append(matrix.get(i, 0));

                // Remaining columns
                for (int j = 1; j < numbCols; j++) {
                    builder.append(",\t").append(matrix.get(i, j));
                }
            }

            // Finish
            builder.append(" } }");
        }

        return builder.toString();
    }

    static Access2D wrap(final double[][] target) {
        return RawStore.wrap(target);
    }

    static > Access2D wrap(final N[][] target) {
        return new Access2D<>() {

            public long count() {
                return Structure2D.count(target.length, target[0].length);
            }

            public long countColumns() {
                return target[0].length;
            }

            public long countRows() {
                return target.length;
            }

            public double doubleValue(final int row, final int col) {
                return NumberDefinition.doubleValue(target[row][col]);
            }

            public double doubleValue(final long row, final long col) {
                return NumberDefinition.doubleValue(target[Math.toIntExact(row)][Math.toIntExact(col)]);
            }

            public N get(final long row, final long col) {
                return target[Math.toIntExact(row)][Math.toIntExact(col)];
            }

            public int getColDim() {
                return target[0].length;
            }

            public int getRowDim() {
                return target.length;
            }

            @Override
            public String toString() {
                return Access2D.toString(this);
            }

        };
    }

    default , R extends Mutate2D.Receiver> Collectable asCollectable2D() {
        return new Collectable<>() {

            public long countColumns() {
                return Access2D.this.countColumns();
            }

            public long countRows() {
                return Access2D.this.countRows();
            }

            public int getColDim() {
                return Access2D.this.getColDim();
            }

            public int getRowDim() {
                return Access2D.this.getRowDim();
            }

            public void supplyTo(final R receiver) {
                receiver.accept(Access2D.this);
            }

        };
    }

    default  Keyed2D asKeyed2D(final IndexMapper rowMapper, final IndexMapper columnMapper) {
        return new Keyed2D<>(this, Structure2D.mapperOf(this, rowMapper, columnMapper));
    }

    @Override
    default byte byteValue(final int index) {
        int structure = this.getRowDim();
        int row = Structure2D.row(index, structure);
        int col = Structure2D.column(index, structure);
        return this.byteValue(row, col);
    }

    default byte byteValue(final int row, final int col) {
        return (byte) this.shortValue(row, col);
    }

    @Override
    default byte byteValue(final long index) {
        long structure = this.countRows();
        long row = Structure2D.row(index, structure);
        long col = Structure2D.column(index, structure);
        return this.byteValue(row, col);
    }

    default byte byteValue(final long row, final long col) {
        return this.byteValue(Math.toIntExact(row), Math.toIntExact(col));
    }

    default ColumnView columns() {
        return new ColumnView<>(this);
    }

    default Access2D columns(final int... columns) {
        return this.select(null, columns);
    }

    default Access2D columns(final long... columns) {
        return this.select(null, columns);
    }

    @Override
    default double doubleValue(final int index) {
        int structure = this.getRowDim();
        int row = Structure2D.row(index, structure);
        int col = Structure2D.column(index, structure);
        return this.doubleValue(row, col);
    }

    /**
     * Extracts one element of this matrix as a double.
     *
     * @param row A row index.
     * @param col A column index.
     * @return One matrix element
     */
    double doubleValue(int row, int col);

    @Override
    default double doubleValue(final long index) {
        long structure = this.countRows();
        long row = Structure2D.row(index, structure);
        long col = Structure2D.column(index, structure);
        return this.doubleValue(row, col);
    }

    default double doubleValue(final long row, final long col) {
        return this.doubleValue(Math.toIntExact(row), Math.toIntExact(col));
    }

    @Override
    default ElementView2D elements() {
        return new Access2D.ElementView<>(Access1D.super.elements(), this.countRows());
    }

    @Override
    default float floatValue(final int index) {
        int structure = this.getRowDim();
        int row = Structure2D.row(index, structure);
        int col = Structure2D.column(index, structure);
        return this.floatValue(row, col);
    }

    default float floatValue(final int row, final int col) {
        return (float) this.doubleValue(row, col);
    }

    @Override
    default float floatValue(final long index) {
        long structure = this.countRows();
        long row = Structure2D.row(index, structure);
        long col = Structure2D.column(index, structure);
        return this.floatValue(row, col);
    }

    default float floatValue(final long row, final long col) {
        return this.floatValue(Math.toIntExact(row), Math.toIntExact(col));
    }

    @Override
    default N get(final long index) {
        long struct = this.countRows();
        long row = Structure2D.row(index, struct);
        long col = Structure2D.column(index, struct);
        return this.get(row, col);
    }

    N get(long row, long col);

    @Override
    default int intValue(final int index) {
        int structure = this.getRowDim();
        int row = Structure2D.row(index, structure);
        int col = Structure2D.column(index, structure);
        return this.intValue(row, col);
    }

    default int intValue(final int row, final int col) {
        return (int) this.longValue(row, col);
    }

    @Override
    default int intValue(final long index) {
        long structure = this.countRows();
        long row = Structure2D.row(index, structure);
        long col = Structure2D.column(index, structure);
        return this.intValue(row, col);
    }

    default int intValue(final long row, final long col) {
        return this.intValue(Math.toIntExact(row), Math.toIntExact(col));
    }

    @Override
    default long longValue(final int index) {
        int structure = this.getRowDim();
        int row = Structure2D.row(index, structure);
        int col = Structure2D.column(index, structure);
        return this.longValue(row, col);
    }

    default long longValue(final int row, final int col) {
        return Math.round(this.doubleValue(row, col));
    }

    @Override
    default long longValue(final long index) {
        long structure = this.countRows();
        long row = Structure2D.row(index, structure);
        long col = Structure2D.column(index, structure);
        return this.longValue(row, col);
    }

    default long longValue(final long row, final long col) {
        return this.longValue(Math.toIntExact(row), Math.toIntExact(col));
    }

    @Override
    default ElementView2D nonzeros() {
        return new Access2D.ElementView<>(Access1D.super.nonzeros(), this.countRows());
    }

    default RowView rows() {
        return new RowView<>(this);
    }

    default Access2D rows(final int... rows) {
        return this.select(rows, null);
    }

    default Access2D rows(final long... rows) {
        return this.select(rows, null);
    }

    default Access2D select(final int[] rows, final int[] columns) {
        return new Access2D.SelectionView<>(this, Structure1D.toLongIndexes(rows), Structure1D.toLongIndexes(columns));
    }

    /**
     * Creates a view of the underlying data structure of only the selected elements. If either the rows or
     * columns input arguments are null or empty arrays, then that transaltes to all rows and/or columns.
     */
    default Access2D select(final long[] rows, final long[] columns) {
        return new Access2D.SelectionView<>(this, rows, columns);
    }

    @Override
    default short shortValue(final int index) {
        int structure = this.getRowDim();
        int row = Structure2D.row(index, structure);
        int col = Structure2D.column(index, structure);
        return this.shortValue(row, col);
    }

    default short shortValue(final int row, final int col) {
        return (short) this.intValue(row, col);
    }

    @Override
    default short shortValue(final long index) {
        long structure = this.countRows();
        long row = Structure2D.row(index, structure);
        long col = Structure2D.column(index, structure);
        return this.shortValue(row, col);
    }

    default short shortValue(final long row, final long col) {
        return this.shortValue(Math.toIntExact(row), Math.toIntExact(col));
    }

    default double[][] toRawCopy2D() {

        int nbRows = this.getRowDim();
        int nbCols = this.getColDim();

        double[][] retVal = new double[nbRows][nbCols];

        double[] tmpRow;
        for (int i = 0; i < nbRows; i++) {
            tmpRow = retVal[i];
            for (int j = 0; j < nbCols; j++) {
                tmpRow[j] = this.doubleValue(i, j);
            }
        }

        return retVal;
    }

}