tech.tablesaw.table.TableSlice Maven / Gradle / Ivy
/*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package tech.tablesaw.table;
import com.google.common.base.Preconditions;
import it.unimi.dsi.fastutil.ints.IntArrays;
import it.unimi.dsi.fastutil.ints.IntComparator;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.PrimitiveIterator;
import java.util.stream.IntStream;
import javax.annotation.Nullable;
import tech.tablesaw.aggregate.NumericAggregateFunction;
import tech.tablesaw.api.NumericColumn;
import tech.tablesaw.api.Row;
import tech.tablesaw.api.Table;
import tech.tablesaw.columns.Column;
import tech.tablesaw.selection.Selection;
import tech.tablesaw.sorting.Sort;
import tech.tablesaw.sorting.SortUtils;
import tech.tablesaw.sorting.comparators.IntComparatorChain;
/**
* A TableSlice is a facade around a Relation that acts as a filter. Requests for data are forwarded
* to the underlying table. A TableSlice can be sorted independently of the underlying table.
*
* A TableSlice is only good until the structure of the underlying table changes.
*/
public class TableSlice extends Relation {
private final Table table;
private String name;
@Nullable private Selection selection;
@Nullable private int[] sortOrder = null;
/**
* Returns a new View constructed from the given table, containing only the rows represented by
* the bitmap
*/
public TableSlice(Table table, Selection rowSelection) {
this.name = table.name();
this.selection = rowSelection;
this.table = table;
}
/**
* Returns a new view constructed from the given table. The view can be sorted independently of
* the table.
*/
public TableSlice(Table table) {
this.name = table.name();
this.selection = null;
this.table = table;
}
@Override
public Column> column(int columnIndex) {
Column> col = table.column(columnIndex);
if (isSorted()) {
return col.subset(sortOrder);
} else if (hasSelection()) {
return col.where(selection);
}
return col;
}
@Override
public Column> column(String columnName) {
return column(table.columnIndex(columnName));
}
@Override
public int columnCount() {
return table.columnCount();
}
@Override
public int rowCount() {
if (hasSelection()) {
return selection.size();
}
return table.rowCount();
}
@Override
public List> columns() {
List> columns = new ArrayList<>();
for (int i = 0; i < columnCount(); i++) {
columns.add(column(i));
}
return columns;
}
@Override
public int columnIndex(Column> column) {
return table.columnIndex(column);
}
@Override
public Object get(int r, int c) {
return table.get(mappedRowNumber(r), c);
}
@Override
public String name() {
return name;
}
public Table getTable() {
return table;
}
/** Clears all rows from this View, leaving the structure in place */
@Override
public void clear() {
sortOrder = null;
selection = Selection.with();
}
/** Removes the sort from this View. */
public void removeSort() {
this.sortOrder = null;
}
/**
* Removes the selection from this view, leaving it with the same number of rows as the underlying
* source table.
*/
public void removeSelection() {
this.selection = null;
}
@Override
public List columnNames() {
return table.columnNames();
}
@Override
public TableSlice addColumns(Column>... column) {
throw new UnsupportedOperationException(
"Class TableSlice does not support the addColumns operation");
}
@Override
public TableSlice removeColumns(Column>... columns) {
throw new UnsupportedOperationException(
"Class TableSlice does not support the removeColumns operation");
}
@Override
public Table first(int nRows) {
int count = 0;
PrimitiveIterator.OfInt it = sourceRowNumberIterator();
Table copy = table.emptyCopy();
while (it.hasNext() && count < nRows) {
int row = it.nextInt();
copy.addRow(table.row(row));
count++;
}
return copy;
}
@Override
public TableSlice setName(String name) {
this.name = name;
return this;
}
public Table asTable() {
Table table = Table.create(this.name());
for (Column> column : this.columns()) {
table.addColumns(column);
}
return table;
}
/**
* IntIterator of source table row numbers that are present in this view. This can be used to in
* combination with the source table to iterate over the cells of a column in a sorted order
* without copying the column.
*
* @return an int iterator of row numbers in the source table that are present in this view.
*/
protected PrimitiveIterator.OfInt sourceRowNumberIterator() {
if (this.isSorted()) {
return Arrays.stream(sortOrder).iterator();
} else if (this.hasSelection()) {
return selection.iterator();
}
return Selection.withRange(0, table.rowCount()).iterator();
}
/**
* Returns the result of applying the given function to the specified column
*
* @param numberColumnName The name of a numeric column in this table
* @param function A numeric reduce function
* @return the function result
* @throws IllegalArgumentException if numberColumnName doesn't name a numeric column in this
* table
*/
public double reduce(String numberColumnName, NumericAggregateFunction function) {
NumericColumn> column = table.numberColumn(numberColumnName);
if (hasSelection()) {
return function.summarize(column.where(selection));
}
return function.summarize(column);
}
/**
* Iterate over the underlying rows in the source table. If you set one of the rows while
* iterating it will change the row in the source table.
*/
@Override
public Iterator iterator() {
return new Iterator() {
private final Row row = new Row(TableSlice.this);
@Override
public Row next() {
return row.next();
}
@Override
public boolean hasNext() {
return row.hasNext();
}
};
}
private boolean hasSelection() {
return selection != null;
}
private boolean isSorted() {
return sortOrder != null;
}
/**
* Maps the view row number to the row number on the underlying source table.
*
* @param rowNumber the row number in the view.
* @return the matching row number in the underlying table.
*/
public int mappedRowNumber(int rowNumber) {
if (isSorted()) {
return sortOrder[rowNumber];
} else if (hasSelection()) {
return selection.get(rowNumber);
}
return rowNumber;
}
/**
* Sort this view in place without modifying or copying the underlying source table. Unlike {@link
* Table#sortOn(Sort)} which returns a copy of the table, this method sorts the view in place.
*
* @param key to sort on.
*/
public void sortOn(Sort key) {
Preconditions.checkArgument(!key.isEmpty());
if (key.size() == 1) {
IntComparator comparator = SortUtils.getComparator(table, key);
this.sortOrder = sortOn(comparator);
} else {
IntComparatorChain chain = SortUtils.getChain(table, key);
this.sortOrder = sortOn(chain);
}
}
/** Returns an array of integers representing the source table indexes in sorted order. */
private int[] sortOn(IntComparator rowComparator) {
int[] newRows;
if (hasSelection()) {
newRows = this.selection.toArray();
} else {
newRows = IntStream.range(0, table.rowCount()).toArray();
}
IntArrays.parallelQuickSort(newRows, rowComparator);
return newRows;
}
}