initial commit
This commit is contained in:
parent
3c9ee26b06
commit
22186c33b9
1458 changed files with 282792 additions and 0 deletions
175
java/src/game/collect/RegularImmutableTable.java
Normal file
175
java/src/game/collect/RegularImmutableTable.java
Normal file
|
@ -0,0 +1,175 @@
|
|||
/*
|
||||
* Copyright (C) 2009 The Guava Authors
|
||||
*
|
||||
* 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 game.collect;
|
||||
|
||||
import static game.collect.Preconditions.checkNotNull;
|
||||
|
||||
import java.util.Collections;
|
||||
import java.util.Comparator;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* An implementation of {@link ImmutableTable} holding an arbitrary number of
|
||||
* cells.
|
||||
*
|
||||
* @author Gregory Kick
|
||||
*/
|
||||
|
||||
abstract class RegularImmutableTable<R, C, V> extends ImmutableTable<R, C, V> {
|
||||
RegularImmutableTable() {}
|
||||
|
||||
abstract Cell<R, C, V> getCell(int iterationIndex);
|
||||
|
||||
@Override
|
||||
final ImmutableSet<Cell<R, C, V>> createCellSet() {
|
||||
return isEmpty() ? ImmutableSet.<Cell<R, C, V>>of() : new CellSet();
|
||||
}
|
||||
|
||||
private final class CellSet extends ImmutableSet<Cell<R, C, V>> {
|
||||
@Override
|
||||
public int size() {
|
||||
return RegularImmutableTable.this.size();
|
||||
}
|
||||
|
||||
@Override
|
||||
public UnmodifiableIterator<Cell<R, C, V>> iterator() {
|
||||
return asList().iterator();
|
||||
}
|
||||
|
||||
@Override
|
||||
ImmutableList<Cell<R, C, V>> createAsList() {
|
||||
return new ImmutableAsList<Cell<R, C, V>>() {
|
||||
@Override
|
||||
public Cell<R, C, V> get(int index) {
|
||||
return getCell(index);
|
||||
}
|
||||
|
||||
@Override
|
||||
ImmutableCollection<Cell<R, C, V>> delegateCollection() {
|
||||
return CellSet.this;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean contains(Object object) {
|
||||
if (object instanceof Cell) {
|
||||
Cell<?, ?, ?> cell = (Cell<?, ?, ?>) object;
|
||||
Object value = get(cell.getRowKey(), cell.getColumnKey());
|
||||
return value != null && value.equals(cell.getValue());
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
boolean isPartialView() {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
abstract V getValue(int iterationIndex);
|
||||
|
||||
@Override
|
||||
final ImmutableCollection<V> createValues() {
|
||||
return isEmpty() ? ImmutableList.<V>of() : new Values();
|
||||
}
|
||||
|
||||
private final class Values extends ImmutableList<V> {
|
||||
@Override
|
||||
public int size() {
|
||||
return RegularImmutableTable.this.size();
|
||||
}
|
||||
|
||||
@Override
|
||||
public V get(int index) {
|
||||
return getValue(index);
|
||||
}
|
||||
|
||||
@Override
|
||||
boolean isPartialView() {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
static <R, C, V> RegularImmutableTable<R, C, V> forCells(
|
||||
List<Cell<R, C, V>> cells,
|
||||
final Comparator<? super R> rowComparator,
|
||||
final Comparator<? super C> columnComparator) {
|
||||
checkNotNull(cells);
|
||||
if (rowComparator != null || columnComparator != null) {
|
||||
/*
|
||||
* This sorting logic leads to a cellSet() ordering that may not be expected and that isn't
|
||||
* documented in the Javadoc. If a row Comparator is provided, cellSet() iterates across the
|
||||
* columns in the first row, the columns in the second row, etc. If a column Comparator is
|
||||
* provided but a row Comparator isn't, cellSet() iterates across the rows in the first
|
||||
* column, the rows in the second column, etc.
|
||||
*/
|
||||
Comparator<Cell<R, C, V>> comparator = new Comparator<Cell<R, C, V>>() {
|
||||
@Override public int compare(Cell<R, C, V> cell1, Cell<R, C, V> cell2) {
|
||||
int rowCompare = (rowComparator == null) ? 0
|
||||
: rowComparator.compare(cell1.getRowKey(), cell2.getRowKey());
|
||||
if (rowCompare != 0) {
|
||||
return rowCompare;
|
||||
}
|
||||
return (columnComparator == null) ? 0
|
||||
: columnComparator.compare(cell1.getColumnKey(), cell2.getColumnKey());
|
||||
}
|
||||
};
|
||||
Collections.sort(cells, comparator);
|
||||
}
|
||||
return forCellsInternal(cells, rowComparator, columnComparator);
|
||||
}
|
||||
|
||||
static <R, C, V> RegularImmutableTable<R, C, V> forCells(
|
||||
Iterable<Cell<R, C, V>> cells) {
|
||||
return forCellsInternal(cells, null, null);
|
||||
}
|
||||
|
||||
/**
|
||||
* A factory that chooses the most space-efficient representation of the
|
||||
* table.
|
||||
*/
|
||||
private static final <R, C, V> RegularImmutableTable<R, C, V>
|
||||
forCellsInternal(Iterable<Cell<R, C, V>> cells,
|
||||
Comparator<? super R> rowComparator,
|
||||
Comparator<? super C> columnComparator) {
|
||||
ImmutableSet.Builder<R> rowSpaceBuilder = ImmutableSet.builder();
|
||||
ImmutableSet.Builder<C> columnSpaceBuilder = ImmutableSet.builder();
|
||||
ImmutableList<Cell<R, C, V>> cellList = ImmutableList.copyOf(cells);
|
||||
for (Cell<R, C, V> cell : cellList) {
|
||||
rowSpaceBuilder.add(cell.getRowKey());
|
||||
columnSpaceBuilder.add(cell.getColumnKey());
|
||||
}
|
||||
|
||||
ImmutableSet<R> rowSpace = rowSpaceBuilder.build();
|
||||
if (rowComparator != null) {
|
||||
List<R> rowList = Lists.newArrayList(rowSpace);
|
||||
Collections.sort(rowList, rowComparator);
|
||||
rowSpace = ImmutableSet.copyOf(rowList);
|
||||
}
|
||||
ImmutableSet<C> columnSpace = columnSpaceBuilder.build();
|
||||
if (columnComparator != null) {
|
||||
List<C> columnList = Lists.newArrayList(columnSpace);
|
||||
Collections.sort(columnList, columnComparator);
|
||||
columnSpace = ImmutableSet.copyOf(columnList);
|
||||
}
|
||||
|
||||
// use a dense table if more than half of the cells have values
|
||||
// TODO(gak): tune this condition based on empirical evidence
|
||||
return (cellList.size() > (((long) rowSpace.size() * columnSpace.size()) / 2)) ?
|
||||
new DenseImmutableTable<R, C, V>(cellList, rowSpace, columnSpace) :
|
||||
new SparseImmutableTable<R, C, V>(cellList, rowSpace, columnSpace);
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue