001/* 002 * Copyright (C) 2008 The Guava Authors 003 * 004 * Licensed under the Apache License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.apache.org/licenses/LICENSE-2.0 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016 017package com.google.common.collect; 018 019import static com.google.common.base.Preconditions.checkArgument; 020import static com.google.common.base.Preconditions.checkNotNull; 021import static com.google.common.collect.NullnessCasts.uncheckedCastNullableTToT; 022 023import com.google.common.annotations.GwtCompatible; 024import com.google.common.annotations.J2ktIncompatible; 025import com.google.common.base.Function; 026import com.google.common.base.Objects; 027import com.google.common.base.Supplier; 028import com.google.common.collect.Table.Cell; 029import java.io.Serializable; 030import java.util.Collection; 031import java.util.Collections; 032import java.util.Iterator; 033import java.util.Map; 034import java.util.Set; 035import java.util.SortedMap; 036import java.util.SortedSet; 037import java.util.function.BinaryOperator; 038import java.util.stream.Collector; 039import javax.annotation.CheckForNull; 040import org.checkerframework.checker.nullness.qual.Nullable; 041 042/** 043 * Provides static methods that involve a {@code Table}. 044 * 045 * <p>See the Guava User Guide article on <a href= 046 * "https://github.com/google/guava/wiki/CollectionUtilitiesExplained#tables">{@code Tables}</a>. 047 * 048 * @author Jared Levy 049 * @author Louis Wasserman 050 * @since 7.0 051 */ 052@GwtCompatible 053@ElementTypesAreNonnullByDefault 054public final class Tables { 055 private Tables() {} 056 057 /** 058 * Returns a {@link Collector} that accumulates elements into a {@code Table} created using the 059 * specified supplier, whose cells are generated by applying the provided mapping functions to the 060 * input elements. Cells are inserted into the generated {@code Table} in encounter order. 061 * 062 * <p>If multiple input elements map to the same row and column, an {@code IllegalStateException} 063 * is thrown when the collection operation is performed. 064 * 065 * <p>To collect to an {@link ImmutableTable}, use {@link ImmutableTable#toImmutableTable}. 066 * 067 * @since 33.2.0 (available since 21.0 in guava-jre) 068 */ 069 @SuppressWarnings({"AndroidJdkLibsChecker", "Java7ApiChecker"}) 070 @IgnoreJRERequirement // Users will use this only if they're already using streams. 071 public static < 072 T extends @Nullable Object, 073 R extends @Nullable Object, 074 C extends @Nullable Object, 075 V, 076 I extends Table<R, C, V>> 077 Collector<T, ?, I> toTable( 078 java.util.function.Function<? super T, ? extends R> rowFunction, 079 java.util.function.Function<? super T, ? extends C> columnFunction, 080 java.util.function.Function<? super T, ? extends V> valueFunction, 081 java.util.function.Supplier<I> tableSupplier) { 082 return TableCollectors.<T, R, C, V, I>toTable( 083 rowFunction, columnFunction, valueFunction, tableSupplier); 084 } 085 086 /** 087 * Returns a {@link Collector} that accumulates elements into a {@code Table} created using the 088 * specified supplier, whose cells are generated by applying the provided mapping functions to the 089 * input elements. Cells are inserted into the generated {@code Table} in encounter order. 090 * 091 * <p>If multiple input elements map to the same row and column, the specified merging function is 092 * used to combine the values. Like {@link 093 * java.util.stream.Collectors#toMap(java.util.function.Function, java.util.function.Function, 094 * BinaryOperator, java.util.function.Supplier)}, this Collector throws a {@code 095 * NullPointerException} on null values returned from {@code valueFunction}, and treats nulls 096 * returned from {@code mergeFunction} as removals of that row/column pair. 097 * 098 * @since 33.2.0 (available since 21.0 in guava-jre) 099 */ 100 @SuppressWarnings({"AndroidJdkLibsChecker", "Java7ApiChecker"}) 101 @IgnoreJRERequirement // Users will use this only if they're already using streams. 102 public static < 103 T extends @Nullable Object, 104 R extends @Nullable Object, 105 C extends @Nullable Object, 106 V, 107 I extends Table<R, C, V>> 108 Collector<T, ?, I> toTable( 109 java.util.function.Function<? super T, ? extends R> rowFunction, 110 java.util.function.Function<? super T, ? extends C> columnFunction, 111 java.util.function.Function<? super T, ? extends V> valueFunction, 112 BinaryOperator<V> mergeFunction, 113 java.util.function.Supplier<I> tableSupplier) { 114 return TableCollectors.<T, R, C, V, I>toTable( 115 rowFunction, columnFunction, valueFunction, mergeFunction, tableSupplier); 116 } 117 118 /** 119 * Returns an immutable cell with the specified row key, column key, and value. 120 * 121 * <p>The returned cell is serializable. 122 * 123 * @param rowKey the row key to be associated with the returned cell 124 * @param columnKey the column key to be associated with the returned cell 125 * @param value the value to be associated with the returned cell 126 */ 127 public static <R extends @Nullable Object, C extends @Nullable Object, V extends @Nullable Object> 128 Cell<R, C, V> immutableCell( 129 @ParametricNullness R rowKey, 130 @ParametricNullness C columnKey, 131 @ParametricNullness V value) { 132 return new ImmutableCell<>(rowKey, columnKey, value); 133 } 134 135 static final class ImmutableCell< 136 R extends @Nullable Object, C extends @Nullable Object, V extends @Nullable Object> 137 extends AbstractCell<R, C, V> implements Serializable { 138 @ParametricNullness private final R rowKey; 139 @ParametricNullness private final C columnKey; 140 @ParametricNullness private final V value; 141 142 ImmutableCell( 143 @ParametricNullness R rowKey, 144 @ParametricNullness C columnKey, 145 @ParametricNullness V value) { 146 this.rowKey = rowKey; 147 this.columnKey = columnKey; 148 this.value = value; 149 } 150 151 @Override 152 @ParametricNullness 153 public R getRowKey() { 154 return rowKey; 155 } 156 157 @Override 158 @ParametricNullness 159 public C getColumnKey() { 160 return columnKey; 161 } 162 163 @Override 164 @ParametricNullness 165 public V getValue() { 166 return value; 167 } 168 169 private static final long serialVersionUID = 0; 170 } 171 172 abstract static class AbstractCell< 173 R extends @Nullable Object, C extends @Nullable Object, V extends @Nullable Object> 174 implements Cell<R, C, V> { 175 // needed for serialization 176 AbstractCell() {} 177 178 @Override 179 public boolean equals(@CheckForNull Object obj) { 180 if (obj == this) { 181 return true; 182 } 183 if (obj instanceof Cell) { 184 Cell<?, ?, ?> other = (Cell<?, ?, ?>) obj; 185 return Objects.equal(getRowKey(), other.getRowKey()) 186 && Objects.equal(getColumnKey(), other.getColumnKey()) 187 && Objects.equal(getValue(), other.getValue()); 188 } 189 return false; 190 } 191 192 @Override 193 public int hashCode() { 194 return Objects.hashCode(getRowKey(), getColumnKey(), getValue()); 195 } 196 197 @Override 198 public String toString() { 199 return "(" + getRowKey() + "," + getColumnKey() + ")=" + getValue(); 200 } 201 } 202 203 /** 204 * Creates a transposed view of a given table that flips its row and column keys. In other words, 205 * calling {@code get(columnKey, rowKey)} on the generated table always returns the same value as 206 * calling {@code get(rowKey, columnKey)} on the original table. Updating the original table 207 * changes the contents of the transposed table and vice versa. 208 * 209 * <p>The returned table supports update operations as long as the input table supports the 210 * analogous operation with swapped rows and columns. For example, in a {@link HashBasedTable} 211 * instance, {@code rowKeySet().iterator()} supports {@code remove()} but {@code 212 * columnKeySet().iterator()} doesn't. With a transposed {@link HashBasedTable}, it's the other 213 * way around. 214 */ 215 public static <R extends @Nullable Object, C extends @Nullable Object, V extends @Nullable Object> 216 Table<C, R, V> transpose(Table<R, C, V> table) { 217 return (table instanceof TransposeTable) 218 ? ((TransposeTable<R, C, V>) table).original 219 : new TransposeTable<C, R, V>(table); 220 } 221 222 private static class TransposeTable< 223 C extends @Nullable Object, R extends @Nullable Object, V extends @Nullable Object> 224 extends AbstractTable<C, R, V> { 225 final Table<R, C, V> original; 226 227 TransposeTable(Table<R, C, V> original) { 228 this.original = checkNotNull(original); 229 } 230 231 @Override 232 public void clear() { 233 original.clear(); 234 } 235 236 @Override 237 public Map<C, V> column(@ParametricNullness R columnKey) { 238 return original.row(columnKey); 239 } 240 241 @Override 242 public Set<R> columnKeySet() { 243 return original.rowKeySet(); 244 } 245 246 @Override 247 public Map<R, Map<C, V>> columnMap() { 248 return original.rowMap(); 249 } 250 251 @Override 252 public boolean contains(@CheckForNull Object rowKey, @CheckForNull Object columnKey) { 253 return original.contains(columnKey, rowKey); 254 } 255 256 @Override 257 public boolean containsColumn(@CheckForNull Object columnKey) { 258 return original.containsRow(columnKey); 259 } 260 261 @Override 262 public boolean containsRow(@CheckForNull Object rowKey) { 263 return original.containsColumn(rowKey); 264 } 265 266 @Override 267 public boolean containsValue(@CheckForNull Object value) { 268 return original.containsValue(value); 269 } 270 271 @Override 272 @CheckForNull 273 public V get(@CheckForNull Object rowKey, @CheckForNull Object columnKey) { 274 return original.get(columnKey, rowKey); 275 } 276 277 @Override 278 @CheckForNull 279 public V put( 280 @ParametricNullness C rowKey, 281 @ParametricNullness R columnKey, 282 @ParametricNullness V value) { 283 return original.put(columnKey, rowKey, value); 284 } 285 286 @Override 287 public void putAll(Table<? extends C, ? extends R, ? extends V> table) { 288 original.putAll(transpose(table)); 289 } 290 291 @Override 292 @CheckForNull 293 public V remove(@CheckForNull Object rowKey, @CheckForNull Object columnKey) { 294 return original.remove(columnKey, rowKey); 295 } 296 297 @Override 298 public Map<R, V> row(@ParametricNullness C rowKey) { 299 return original.column(rowKey); 300 } 301 302 @Override 303 public Set<C> rowKeySet() { 304 return original.columnKeySet(); 305 } 306 307 @Override 308 public Map<C, Map<R, V>> rowMap() { 309 return original.columnMap(); 310 } 311 312 @Override 313 public int size() { 314 return original.size(); 315 } 316 317 @Override 318 public Collection<V> values() { 319 return original.values(); 320 } 321 322 @Override 323 Iterator<Cell<C, R, V>> cellIterator() { 324 return Iterators.transform(original.cellSet().iterator(), Tables::transposeCell); 325 } 326 } 327 328 private static < 329 R extends @Nullable Object, C extends @Nullable Object, V extends @Nullable Object> 330 Cell<C, R, V> transposeCell(Cell<R, C, V> cell) { 331 return immutableCell(cell.getColumnKey(), cell.getRowKey(), cell.getValue()); 332 } 333 334 /** 335 * Creates a table that uses the specified backing map and factory. It can generate a table based 336 * on arbitrary {@link Map} classes. 337 * 338 * <p>The {@code factory}-generated and {@code backingMap} classes determine the table iteration 339 * order. However, the table's {@code row()} method returns instances of a different class than 340 * {@code factory.get()} does. 341 * 342 * <p>Call this method only when the simpler factory methods in classes like {@link 343 * HashBasedTable} and {@link TreeBasedTable} won't suffice. 344 * 345 * <p>The views returned by the {@code Table} methods {@link Table#column}, {@link 346 * Table#columnKeySet}, and {@link Table#columnMap} have iterators that don't support {@code 347 * remove()}. Otherwise, all optional operations are supported. Null row keys, columns keys, and 348 * values are not supported. 349 * 350 * <p>Lookups by row key are often faster than lookups by column key, because the data is stored 351 * in a {@code Map<R, Map<C, V>>}. A method call like {@code column(columnKey).get(rowKey)} still 352 * runs quickly, since the row key is provided. However, {@code column(columnKey).size()} takes 353 * longer, since an iteration across all row keys occurs. 354 * 355 * <p>Note that this implementation is not synchronized. If multiple threads access this table 356 * concurrently and one of the threads modifies the table, it must be synchronized externally. 357 * 358 * <p>The table is serializable if {@code backingMap}, {@code factory}, the maps generated by 359 * {@code factory}, and the table contents are all serializable. 360 * 361 * <p>Note: the table assumes complete ownership over of {@code backingMap} and the maps returned 362 * by {@code factory}. Those objects should not be manually updated and they should not use soft, 363 * weak, or phantom references. 364 * 365 * @param backingMap place to store the mapping from each row key to its corresponding column key 366 * / value map 367 * @param factory supplier of new, empty maps that will each hold all column key / value mappings 368 * for a given row key 369 * @throws IllegalArgumentException if {@code backingMap} is not empty 370 * @since 10.0 371 */ 372 public static <R, C, V> Table<R, C, V> newCustomTable( 373 Map<R, Map<C, V>> backingMap, Supplier<? extends Map<C, V>> factory) { 374 checkArgument(backingMap.isEmpty()); 375 checkNotNull(factory); 376 // TODO(jlevy): Wrap factory to validate that the supplied maps are empty? 377 return new StandardTable<>(backingMap, factory); 378 } 379 380 /** 381 * Returns a view of a table where each value is transformed by a function. All other properties 382 * of the table, such as iteration order, are left intact. 383 * 384 * <p>Changes in the underlying table are reflected in this view. Conversely, this view supports 385 * removal operations, and these are reflected in the underlying table. 386 * 387 * <p>It's acceptable for the underlying table to contain null keys, and even null values provided 388 * that the function is capable of accepting null input. The transformed table might contain null 389 * values, if the function sometimes gives a null result. 390 * 391 * <p>The returned table is not thread-safe or serializable, even if the underlying table is. 392 * 393 * <p>The function is applied lazily, invoked when needed. This is necessary for the returned 394 * table to be a view, but it means that the function will be applied many times for bulk 395 * operations like {@link Table#containsValue} and {@code Table.toString()}. For this to perform 396 * well, {@code function} should be fast. To avoid lazy evaluation when the returned table doesn't 397 * need to be a view, copy the returned table into a new table of your choosing. 398 * 399 * @since 10.0 400 */ 401 public static < 402 R extends @Nullable Object, 403 C extends @Nullable Object, 404 V1 extends @Nullable Object, 405 V2 extends @Nullable Object> 406 Table<R, C, V2> transformValues( 407 Table<R, C, V1> fromTable, Function<? super V1, V2> function) { 408 return new TransformedTable<>(fromTable, function); 409 } 410 411 private static class TransformedTable< 412 R extends @Nullable Object, 413 C extends @Nullable Object, 414 V1 extends @Nullable Object, 415 V2 extends @Nullable Object> 416 extends AbstractTable<R, C, V2> { 417 final Table<R, C, V1> fromTable; 418 final Function<? super V1, V2> function; 419 420 TransformedTable(Table<R, C, V1> fromTable, Function<? super V1, V2> function) { 421 this.fromTable = checkNotNull(fromTable); 422 this.function = checkNotNull(function); 423 } 424 425 @Override 426 public boolean contains(@CheckForNull Object rowKey, @CheckForNull Object columnKey) { 427 return fromTable.contains(rowKey, columnKey); 428 } 429 430 @Override 431 @CheckForNull 432 public V2 get(@CheckForNull Object rowKey, @CheckForNull Object columnKey) { 433 // The function is passed a null input only when the table contains a null 434 // value. 435 // The cast is safe because of the contains() check. 436 return contains(rowKey, columnKey) 437 ? function.apply(uncheckedCastNullableTToT(fromTable.get(rowKey, columnKey))) 438 : null; 439 } 440 441 @Override 442 public int size() { 443 return fromTable.size(); 444 } 445 446 @Override 447 public void clear() { 448 fromTable.clear(); 449 } 450 451 @Override 452 @CheckForNull 453 public V2 put( 454 @ParametricNullness R rowKey, 455 @ParametricNullness C columnKey, 456 @ParametricNullness V2 value) { 457 throw new UnsupportedOperationException(); 458 } 459 460 @Override 461 public void putAll(Table<? extends R, ? extends C, ? extends V2> table) { 462 throw new UnsupportedOperationException(); 463 } 464 465 @Override 466 @CheckForNull 467 public V2 remove(@CheckForNull Object rowKey, @CheckForNull Object columnKey) { 468 return contains(rowKey, columnKey) 469 // The cast is safe because of the contains() check. 470 ? function.apply(uncheckedCastNullableTToT(fromTable.remove(rowKey, columnKey))) 471 : null; 472 } 473 474 @Override 475 public Map<C, V2> row(@ParametricNullness R rowKey) { 476 return Maps.transformValues(fromTable.row(rowKey), function); 477 } 478 479 @Override 480 public Map<R, V2> column(@ParametricNullness C columnKey) { 481 return Maps.transformValues(fromTable.column(columnKey), function); 482 } 483 484 Function<Cell<R, C, V1>, Cell<R, C, V2>> cellFunction() { 485 return new Function<Cell<R, C, V1>, Cell<R, C, V2>>() { 486 @Override 487 public Cell<R, C, V2> apply(Cell<R, C, V1> cell) { 488 return immutableCell( 489 cell.getRowKey(), cell.getColumnKey(), function.apply(cell.getValue())); 490 } 491 }; 492 } 493 494 @Override 495 Iterator<Cell<R, C, V2>> cellIterator() { 496 return Iterators.transform(fromTable.cellSet().iterator(), cellFunction()); 497 } 498 499 @Override 500 public Set<R> rowKeySet() { 501 return fromTable.rowKeySet(); 502 } 503 504 @Override 505 public Set<C> columnKeySet() { 506 return fromTable.columnKeySet(); 507 } 508 509 @Override 510 Collection<V2> createValues() { 511 return Collections2.transform(fromTable.values(), function); 512 } 513 514 @Override 515 public Map<R, Map<C, V2>> rowMap() { 516 Function<Map<C, V1>, Map<C, V2>> rowFunction = 517 new Function<Map<C, V1>, Map<C, V2>>() { 518 @Override 519 public Map<C, V2> apply(Map<C, V1> row) { 520 return Maps.transformValues(row, function); 521 } 522 }; 523 return Maps.transformValues(fromTable.rowMap(), rowFunction); 524 } 525 526 @Override 527 public Map<C, Map<R, V2>> columnMap() { 528 Function<Map<R, V1>, Map<R, V2>> columnFunction = 529 new Function<Map<R, V1>, Map<R, V2>>() { 530 @Override 531 public Map<R, V2> apply(Map<R, V1> column) { 532 return Maps.transformValues(column, function); 533 } 534 }; 535 return Maps.transformValues(fromTable.columnMap(), columnFunction); 536 } 537 } 538 539 /** 540 * Returns an unmodifiable view of the specified table. This method allows modules to provide 541 * users with "read-only" access to internal tables. Query operations on the returned table "read 542 * through" to the specified table, and attempts to modify the returned table, whether direct or 543 * via its collection views, result in an {@code UnsupportedOperationException}. 544 * 545 * <p>The returned table will be serializable if the specified table is serializable. 546 * 547 * <p>Consider using an {@link ImmutableTable}, which is guaranteed never to change. 548 * 549 * @since 11.0 550 */ 551 public static <R extends @Nullable Object, C extends @Nullable Object, V extends @Nullable Object> 552 Table<R, C, V> unmodifiableTable(Table<? extends R, ? extends C, ? extends V> table) { 553 return new UnmodifiableTable<>(table); 554 } 555 556 private static class UnmodifiableTable< 557 R extends @Nullable Object, C extends @Nullable Object, V extends @Nullable Object> 558 extends ForwardingTable<R, C, V> implements Serializable { 559 final Table<? extends R, ? extends C, ? extends V> delegate; 560 561 UnmodifiableTable(Table<? extends R, ? extends C, ? extends V> delegate) { 562 this.delegate = checkNotNull(delegate); 563 } 564 565 @SuppressWarnings("unchecked") // safe, covariant cast 566 @Override 567 protected Table<R, C, V> delegate() { 568 return (Table<R, C, V>) delegate; 569 } 570 571 @Override 572 public Set<Cell<R, C, V>> cellSet() { 573 return Collections.unmodifiableSet(super.cellSet()); 574 } 575 576 @Override 577 public void clear() { 578 throw new UnsupportedOperationException(); 579 } 580 581 @Override 582 public Map<R, V> column(@ParametricNullness C columnKey) { 583 return Collections.unmodifiableMap(super.column(columnKey)); 584 } 585 586 @Override 587 public Set<C> columnKeySet() { 588 return Collections.unmodifiableSet(super.columnKeySet()); 589 } 590 591 @Override 592 public Map<C, Map<R, V>> columnMap() { 593 Function<Map<R, V>, Map<R, V>> wrapper = unmodifiableWrapper(); 594 return Collections.unmodifiableMap(Maps.transformValues(super.columnMap(), wrapper)); 595 } 596 597 @Override 598 @CheckForNull 599 public V put( 600 @ParametricNullness R rowKey, 601 @ParametricNullness C columnKey, 602 @ParametricNullness V value) { 603 throw new UnsupportedOperationException(); 604 } 605 606 @Override 607 public void putAll(Table<? extends R, ? extends C, ? extends V> table) { 608 throw new UnsupportedOperationException(); 609 } 610 611 @Override 612 @CheckForNull 613 public V remove(@CheckForNull Object rowKey, @CheckForNull Object columnKey) { 614 throw new UnsupportedOperationException(); 615 } 616 617 @Override 618 public Map<C, V> row(@ParametricNullness R rowKey) { 619 return Collections.unmodifiableMap(super.row(rowKey)); 620 } 621 622 @Override 623 public Set<R> rowKeySet() { 624 return Collections.unmodifiableSet(super.rowKeySet()); 625 } 626 627 @Override 628 public Map<R, Map<C, V>> rowMap() { 629 Function<Map<C, V>, Map<C, V>> wrapper = unmodifiableWrapper(); 630 return Collections.unmodifiableMap(Maps.transformValues(super.rowMap(), wrapper)); 631 } 632 633 @Override 634 public Collection<V> values() { 635 return Collections.unmodifiableCollection(super.values()); 636 } 637 638 private static final long serialVersionUID = 0; 639 } 640 641 /** 642 * Returns an unmodifiable view of the specified row-sorted table. This method allows modules to 643 * provide users with "read-only" access to internal tables. Query operations on the returned 644 * table "read through" to the specified table, and attempts to modify the returned table, whether 645 * direct or via its collection views, result in an {@code UnsupportedOperationException}. 646 * 647 * <p>The returned table will be serializable if the specified table is serializable. 648 * 649 * @param table the row-sorted table for which an unmodifiable view is to be returned 650 * @return an unmodifiable view of the specified table 651 * @since 11.0 652 */ 653 public static <R extends @Nullable Object, C extends @Nullable Object, V extends @Nullable Object> 654 RowSortedTable<R, C, V> unmodifiableRowSortedTable( 655 RowSortedTable<R, ? extends C, ? extends V> table) { 656 /* 657 * It's not ? extends R, because it's technically not covariant in R. Specifically, 658 * table.rowMap().comparator() could return a comparator that only works for the ? extends R. 659 * Collections.unmodifiableSortedMap makes the same distinction. 660 */ 661 return new UnmodifiableRowSortedMap<>(table); 662 } 663 664 private static final class UnmodifiableRowSortedMap< 665 R extends @Nullable Object, C extends @Nullable Object, V extends @Nullable Object> 666 extends UnmodifiableTable<R, C, V> implements RowSortedTable<R, C, V> { 667 668 public UnmodifiableRowSortedMap(RowSortedTable<R, ? extends C, ? extends V> delegate) { 669 super(delegate); 670 } 671 672 @Override 673 protected RowSortedTable<R, C, V> delegate() { 674 return (RowSortedTable<R, C, V>) super.delegate(); 675 } 676 677 @Override 678 public SortedMap<R, Map<C, V>> rowMap() { 679 Function<Map<C, V>, Map<C, V>> wrapper = unmodifiableWrapper(); 680 return Collections.unmodifiableSortedMap(Maps.transformValues(delegate().rowMap(), wrapper)); 681 } 682 683 @Override 684 public SortedSet<R> rowKeySet() { 685 return Collections.unmodifiableSortedSet(delegate().rowKeySet()); 686 } 687 688 private static final long serialVersionUID = 0; 689 } 690 691 @SuppressWarnings("unchecked") 692 private static <K extends @Nullable Object, V extends @Nullable Object> 693 Function<Map<K, V>, Map<K, V>> unmodifiableWrapper() { 694 return (Function) UNMODIFIABLE_WRAPPER; 695 } 696 697 private static final Function<? extends Map<?, ?>, ? extends Map<?, ?>> UNMODIFIABLE_WRAPPER = 698 new Function<Map<Object, Object>, Map<Object, Object>>() { 699 @Override 700 public Map<Object, Object> apply(Map<Object, Object> input) { 701 return Collections.unmodifiableMap(input); 702 } 703 }; 704 705 /** 706 * Returns a synchronized (thread-safe) table backed by the specified table. In order to guarantee 707 * serial access, it is critical that <b>all</b> access to the backing table is accomplished 708 * through the returned table. 709 * 710 * <p>It is imperative that the user manually synchronize on the returned table when accessing any 711 * of its collection views: 712 * 713 * <pre>{@code 714 * Table<R, C, V> table = Tables.synchronizedTable(HashBasedTable.<R, C, V>create()); 715 * ... 716 * Map<C, V> row = table.row(rowKey); // Needn't be in synchronized block 717 * ... 718 * synchronized (table) { // Synchronizing on table, not row! 719 * Iterator<Entry<C, V>> i = row.entrySet().iterator(); // Must be in synchronized block 720 * while (i.hasNext()) { 721 * foo(i.next()); 722 * } 723 * } 724 * }</pre> 725 * 726 * <p>Failure to follow this advice may result in non-deterministic behavior. 727 * 728 * <p>The returned table will be serializable if the specified table is serializable. 729 * 730 * @param table the table to be wrapped in a synchronized view 731 * @return a synchronized view of the specified table 732 * @since 22.0 733 */ 734 @J2ktIncompatible // Synchronized 735 public static <R extends @Nullable Object, C extends @Nullable Object, V extends @Nullable Object> 736 Table<R, C, V> synchronizedTable(Table<R, C, V> table) { 737 return Synchronized.table(table, null); 738 } 739 740 static boolean equalsImpl(Table<?, ?, ?> table, @CheckForNull Object obj) { 741 if (obj == table) { 742 return true; 743 } else if (obj instanceof Table) { 744 Table<?, ?, ?> that = (Table<?, ?, ?>) obj; 745 return table.cellSet().equals(that.cellSet()); 746 } else { 747 return false; 748 } 749 } 750}