1 /* 2 * Copyright (C) 2007 The Guava Authors 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.google.common.collect; 18 19 import static com.google.common.base.Preconditions.checkArgument; 20 import static com.google.common.base.Preconditions.checkState; 21 import static com.google.common.collect.CollectPreconditions.checkNonnegative; 22 import static com.google.common.collect.CollectPreconditions.checkRemove; 23 24 import com.google.common.annotations.GwtCompatible; 25 import com.google.common.annotations.GwtIncompatible; 26 import com.google.common.base.MoreObjects; 27 import com.google.common.primitives.Ints; 28 29 import java.io.IOException; 30 import java.io.ObjectInputStream; 31 import java.io.ObjectOutputStream; 32 import java.io.Serializable; 33 import java.util.Comparator; 34 import java.util.ConcurrentModificationException; 35 import java.util.Iterator; 36 import java.util.NoSuchElementException; 37 38 import javax.annotation.Nullable; 39 40 /** 41 * A multiset which maintains the ordering of its elements, according to either their natural order 42 * or an explicit {@link Comparator}. In all cases, this implementation uses 43 * {@link Comparable#compareTo} or {@link Comparator#compare} instead of {@link Object#equals} to 44 * determine equivalence of instances. 45 * 46 * <p><b>Warning:</b> The comparison must be <i>consistent with equals</i> as explained by the 47 * {@link Comparable} class specification. Otherwise, the resulting multiset will violate the 48 * {@link java.util.Collection} contract, which is specified in terms of {@link Object#equals}. 49 * 50 * <p>See the Guava User Guide article on <a href= 51 * "http://code.google.com/p/guava-libraries/wiki/NewCollectionTypesExplained#Multiset"> 52 * {@code Multiset}</a>. 53 * 54 * @author Louis Wasserman 55 * @author Jared Levy 56 * @since 2.0 (imported from Google Collections Library) 57 */ 58 @GwtCompatible(emulated = true) 59 public final class TreeMultiset<E> extends AbstractSortedMultiset<E> implements Serializable { 60 61 /** 62 * Creates a new, empty multiset, sorted according to the elements' natural order. All elements 63 * inserted into the multiset must implement the {@code Comparable} interface. Furthermore, all 64 * such elements must be <i>mutually comparable</i>: {@code e1.compareTo(e2)} must not throw a 65 * {@code ClassCastException} for any elements {@code e1} and {@code e2} in the multiset. If the 66 * user attempts to add an element to the multiset that violates this constraint (for example, 67 * the user attempts to add a string element to a set whose elements are integers), the 68 * {@code add(Object)} call will throw a {@code ClassCastException}. 69 * 70 * <p>The type specification is {@code <E extends Comparable>}, instead of the more specific 71 * {@code <E extends Comparable<? super E>>}, to support classes defined without generics. 72 */ create()73 public static <E extends Comparable> TreeMultiset<E> create() { 74 return new TreeMultiset<E>(Ordering.natural()); 75 } 76 77 /** 78 * Creates a new, empty multiset, sorted according to the specified comparator. All elements 79 * inserted into the multiset must be <i>mutually comparable</i> by the specified comparator: 80 * {@code comparator.compare(e1, 81 * e2)} must not throw a {@code ClassCastException} for any elements {@code e1} and {@code e2} in 82 * the multiset. If the user attempts to add an element to the multiset that violates this 83 * constraint, the {@code add(Object)} call will throw a {@code ClassCastException}. 84 * 85 * @param comparator 86 * the comparator that will be used to sort this multiset. A null value indicates that 87 * the elements' <i>natural ordering</i> should be used. 88 */ 89 @SuppressWarnings("unchecked") create(@ullable Comparator<? super E> comparator)90 public static <E> TreeMultiset<E> create(@Nullable Comparator<? super E> comparator) { 91 return (comparator == null) 92 ? new TreeMultiset<E>((Comparator) Ordering.natural()) 93 : new TreeMultiset<E>(comparator); 94 } 95 96 /** 97 * Creates an empty multiset containing the given initial elements, sorted according to the 98 * elements' natural order. 99 * 100 * <p>This implementation is highly efficient when {@code elements} is itself a {@link Multiset}. 101 * 102 * <p>The type specification is {@code <E extends Comparable>}, instead of the more specific 103 * {@code <E extends Comparable<? super E>>}, to support classes defined without generics. 104 */ create(Iterable<? extends E> elements)105 public static <E extends Comparable> TreeMultiset<E> create(Iterable<? extends E> elements) { 106 TreeMultiset<E> multiset = create(); 107 Iterables.addAll(multiset, elements); 108 return multiset; 109 } 110 111 private final transient Reference<AvlNode<E>> rootReference; 112 private final transient GeneralRange<E> range; 113 private final transient AvlNode<E> header; 114 TreeMultiset(Reference<AvlNode<E>> rootReference, GeneralRange<E> range, AvlNode<E> endLink)115 TreeMultiset(Reference<AvlNode<E>> rootReference, GeneralRange<E> range, AvlNode<E> endLink) { 116 super(range.comparator()); 117 this.rootReference = rootReference; 118 this.range = range; 119 this.header = endLink; 120 } 121 TreeMultiset(Comparator<? super E> comparator)122 TreeMultiset(Comparator<? super E> comparator) { 123 super(comparator); 124 this.range = GeneralRange.all(comparator); 125 this.header = new AvlNode<E>(null, 1); 126 successor(header, header); 127 this.rootReference = new Reference<AvlNode<E>>(); 128 } 129 130 /** 131 * A function which can be summed across a subtree. 132 */ 133 private enum Aggregate { 134 SIZE { 135 @Override nodeAggregate(AvlNode<?> node)136 int nodeAggregate(AvlNode<?> node) { 137 return node.elemCount; 138 } 139 140 @Override treeAggregate(@ullable AvlNode<?> root)141 long treeAggregate(@Nullable AvlNode<?> root) { 142 return (root == null) ? 0 : root.totalCount; 143 } 144 }, 145 DISTINCT { 146 @Override nodeAggregate(AvlNode<?> node)147 int nodeAggregate(AvlNode<?> node) { 148 return 1; 149 } 150 151 @Override treeAggregate(@ullable AvlNode<?> root)152 long treeAggregate(@Nullable AvlNode<?> root) { 153 return (root == null) ? 0 : root.distinctElements; 154 } 155 }; nodeAggregate(AvlNode<?> node)156 abstract int nodeAggregate(AvlNode<?> node); 157 treeAggregate(@ullable AvlNode<?> root)158 abstract long treeAggregate(@Nullable AvlNode<?> root); 159 } 160 aggregateForEntries(Aggregate aggr)161 private long aggregateForEntries(Aggregate aggr) { 162 AvlNode<E> root = rootReference.get(); 163 long total = aggr.treeAggregate(root); 164 if (range.hasLowerBound()) { 165 total -= aggregateBelowRange(aggr, root); 166 } 167 if (range.hasUpperBound()) { 168 total -= aggregateAboveRange(aggr, root); 169 } 170 return total; 171 } 172 aggregateBelowRange(Aggregate aggr, @Nullable AvlNode<E> node)173 private long aggregateBelowRange(Aggregate aggr, @Nullable AvlNode<E> node) { 174 if (node == null) { 175 return 0; 176 } 177 int cmp = comparator().compare(range.getLowerEndpoint(), node.elem); 178 if (cmp < 0) { 179 return aggregateBelowRange(aggr, node.left); 180 } else if (cmp == 0) { 181 switch (range.getLowerBoundType()) { 182 case OPEN: 183 return aggr.nodeAggregate(node) + aggr.treeAggregate(node.left); 184 case CLOSED: 185 return aggr.treeAggregate(node.left); 186 default: 187 throw new AssertionError(); 188 } 189 } else { 190 return aggr.treeAggregate(node.left) + aggr.nodeAggregate(node) 191 + aggregateBelowRange(aggr, node.right); 192 } 193 } 194 aggregateAboveRange(Aggregate aggr, @Nullable AvlNode<E> node)195 private long aggregateAboveRange(Aggregate aggr, @Nullable AvlNode<E> node) { 196 if (node == null) { 197 return 0; 198 } 199 int cmp = comparator().compare(range.getUpperEndpoint(), node.elem); 200 if (cmp > 0) { 201 return aggregateAboveRange(aggr, node.right); 202 } else if (cmp == 0) { 203 switch (range.getUpperBoundType()) { 204 case OPEN: 205 return aggr.nodeAggregate(node) + aggr.treeAggregate(node.right); 206 case CLOSED: 207 return aggr.treeAggregate(node.right); 208 default: 209 throw new AssertionError(); 210 } 211 } else { 212 return aggr.treeAggregate(node.right) + aggr.nodeAggregate(node) 213 + aggregateAboveRange(aggr, node.left); 214 } 215 } 216 217 @Override size()218 public int size() { 219 return Ints.saturatedCast(aggregateForEntries(Aggregate.SIZE)); 220 } 221 222 @Override distinctElements()223 int distinctElements() { 224 return Ints.saturatedCast(aggregateForEntries(Aggregate.DISTINCT)); 225 } 226 227 @Override count(@ullable Object element)228 public int count(@Nullable Object element) { 229 try { 230 @SuppressWarnings("unchecked") 231 E e = (E) element; 232 AvlNode<E> root = rootReference.get(); 233 if (!range.contains(e) || root == null) { 234 return 0; 235 } 236 return root.count(comparator(), e); 237 } catch (ClassCastException e) { 238 return 0; 239 } catch (NullPointerException e) { 240 return 0; 241 } 242 } 243 244 @Override add(@ullable E element, int occurrences)245 public int add(@Nullable E element, int occurrences) { 246 checkNonnegative(occurrences, "occurrences"); 247 if (occurrences == 0) { 248 return count(element); 249 } 250 checkArgument(range.contains(element)); 251 AvlNode<E> root = rootReference.get(); 252 if (root == null) { 253 comparator().compare(element, element); 254 AvlNode<E> newRoot = new AvlNode<E>(element, occurrences); 255 successor(header, newRoot, header); 256 rootReference.checkAndSet(root, newRoot); 257 return 0; 258 } 259 int[] result = new int[1]; // used as a mutable int reference to hold result 260 AvlNode<E> newRoot = root.add(comparator(), element, occurrences, result); 261 rootReference.checkAndSet(root, newRoot); 262 return result[0]; 263 } 264 265 @Override remove(@ullable Object element, int occurrences)266 public int remove(@Nullable Object element, int occurrences) { 267 checkNonnegative(occurrences, "occurrences"); 268 if (occurrences == 0) { 269 return count(element); 270 } 271 AvlNode<E> root = rootReference.get(); 272 int[] result = new int[1]; // used as a mutable int reference to hold result 273 AvlNode<E> newRoot; 274 try { 275 @SuppressWarnings("unchecked") 276 E e = (E) element; 277 if (!range.contains(e) || root == null) { 278 return 0; 279 } 280 newRoot = root.remove(comparator(), e, occurrences, result); 281 } catch (ClassCastException e) { 282 return 0; 283 } catch (NullPointerException e) { 284 return 0; 285 } 286 rootReference.checkAndSet(root, newRoot); 287 return result[0]; 288 } 289 290 @Override setCount(@ullable E element, int count)291 public int setCount(@Nullable E element, int count) { 292 checkNonnegative(count, "count"); 293 if (!range.contains(element)) { 294 checkArgument(count == 0); 295 return 0; 296 } 297 298 AvlNode<E> root = rootReference.get(); 299 if (root == null) { 300 if (count > 0) { 301 add(element, count); 302 } 303 return 0; 304 } 305 int[] result = new int[1]; // used as a mutable int reference to hold result 306 AvlNode<E> newRoot = root.setCount(comparator(), element, count, result); 307 rootReference.checkAndSet(root, newRoot); 308 return result[0]; 309 } 310 311 @Override setCount(@ullable E element, int oldCount, int newCount)312 public boolean setCount(@Nullable E element, int oldCount, int newCount) { 313 checkNonnegative(newCount, "newCount"); 314 checkNonnegative(oldCount, "oldCount"); 315 checkArgument(range.contains(element)); 316 317 AvlNode<E> root = rootReference.get(); 318 if (root == null) { 319 if (oldCount == 0) { 320 if (newCount > 0) { 321 add(element, newCount); 322 } 323 return true; 324 } else { 325 return false; 326 } 327 } 328 int[] result = new int[1]; // used as a mutable int reference to hold result 329 AvlNode<E> newRoot = root.setCount(comparator(), element, oldCount, newCount, result); 330 rootReference.checkAndSet(root, newRoot); 331 return result[0] == oldCount; 332 } 333 wrapEntry(final AvlNode<E> baseEntry)334 private Entry<E> wrapEntry(final AvlNode<E> baseEntry) { 335 return new Multisets.AbstractEntry<E>() { 336 @Override 337 public E getElement() { 338 return baseEntry.getElement(); 339 } 340 341 @Override 342 public int getCount() { 343 int result = baseEntry.getCount(); 344 if (result == 0) { 345 return count(getElement()); 346 } else { 347 return result; 348 } 349 } 350 }; 351 } 352 353 /** 354 * Returns the first node in the tree that is in range. 355 */ 356 @Nullable private AvlNode<E> firstNode() { 357 AvlNode<E> root = rootReference.get(); 358 if (root == null) { 359 return null; 360 } 361 AvlNode<E> node; 362 if (range.hasLowerBound()) { 363 E endpoint = range.getLowerEndpoint(); 364 node = rootReference.get().ceiling(comparator(), endpoint); 365 if (node == null) { 366 return null; 367 } 368 if (range.getLowerBoundType() == BoundType.OPEN 369 && comparator().compare(endpoint, node.getElement()) == 0) { 370 node = node.succ; 371 } 372 } else { 373 node = header.succ; 374 } 375 return (node == header || !range.contains(node.getElement())) ? null : node; 376 } 377 378 @Nullable private AvlNode<E> lastNode() { 379 AvlNode<E> root = rootReference.get(); 380 if (root == null) { 381 return null; 382 } 383 AvlNode<E> node; 384 if (range.hasUpperBound()) { 385 E endpoint = range.getUpperEndpoint(); 386 node = rootReference.get().floor(comparator(), endpoint); 387 if (node == null) { 388 return null; 389 } 390 if (range.getUpperBoundType() == BoundType.OPEN 391 && comparator().compare(endpoint, node.getElement()) == 0) { 392 node = node.pred; 393 } 394 } else { 395 node = header.pred; 396 } 397 return (node == header || !range.contains(node.getElement())) ? null : node; 398 } 399 400 @Override 401 Iterator<Entry<E>> entryIterator() { 402 return new Iterator<Entry<E>>() { 403 AvlNode<E> current = firstNode(); 404 Entry<E> prevEntry; 405 406 @Override 407 public boolean hasNext() { 408 if (current == null) { 409 return false; 410 } else if (range.tooHigh(current.getElement())) { 411 current = null; 412 return false; 413 } else { 414 return true; 415 } 416 } 417 418 @Override 419 public Entry<E> next() { 420 if (!hasNext()) { 421 throw new NoSuchElementException(); 422 } 423 Entry<E> result = wrapEntry(current); 424 prevEntry = result; 425 if (current.succ == header) { 426 current = null; 427 } else { 428 current = current.succ; 429 } 430 return result; 431 } 432 433 @Override 434 public void remove() { 435 checkRemove(prevEntry != null); 436 setCount(prevEntry.getElement(), 0); 437 prevEntry = null; 438 } 439 }; 440 } 441 442 @Override 443 Iterator<Entry<E>> descendingEntryIterator() { 444 return new Iterator<Entry<E>>() { 445 AvlNode<E> current = lastNode(); 446 Entry<E> prevEntry = null; 447 448 @Override 449 public boolean hasNext() { 450 if (current == null) { 451 return false; 452 } else if (range.tooLow(current.getElement())) { 453 current = null; 454 return false; 455 } else { 456 return true; 457 } 458 } 459 460 @Override 461 public Entry<E> next() { 462 if (!hasNext()) { 463 throw new NoSuchElementException(); 464 } 465 Entry<E> result = wrapEntry(current); 466 prevEntry = result; 467 if (current.pred == header) { 468 current = null; 469 } else { 470 current = current.pred; 471 } 472 return result; 473 } 474 475 @Override 476 public void remove() { 477 checkRemove(prevEntry != null); 478 setCount(prevEntry.getElement(), 0); 479 prevEntry = null; 480 } 481 }; 482 } 483 484 @Override 485 public SortedMultiset<E> headMultiset(@Nullable E upperBound, BoundType boundType) { 486 return new TreeMultiset<E>(rootReference, range.intersect(GeneralRange.upTo( 487 comparator(), 488 upperBound, 489 boundType)), header); 490 } 491 492 @Override 493 public SortedMultiset<E> tailMultiset(@Nullable E lowerBound, BoundType boundType) { 494 return new TreeMultiset<E>(rootReference, range.intersect(GeneralRange.downTo( 495 comparator(), 496 lowerBound, 497 boundType)), header); 498 } 499 500 static int distinctElements(@Nullable AvlNode<?> node) { 501 return (node == null) ? 0 : node.distinctElements; 502 } 503 504 private static final class Reference<T> { 505 @Nullable private T value; 506 507 @Nullable public T get() { 508 return value; 509 } 510 511 public void checkAndSet(@Nullable T expected, T newValue) { 512 if (value != expected) { 513 throw new ConcurrentModificationException(); 514 } 515 value = newValue; 516 } 517 } 518 519 private static final class AvlNode<E> extends Multisets.AbstractEntry<E> { 520 @Nullable private final E elem; 521 522 // elemCount is 0 iff this node has been deleted. 523 private int elemCount; 524 525 private int distinctElements; 526 private long totalCount; 527 private int height; 528 private AvlNode<E> left; 529 private AvlNode<E> right; 530 private AvlNode<E> pred; 531 private AvlNode<E> succ; 532 533 AvlNode(@Nullable E elem, int elemCount) { 534 checkArgument(elemCount > 0); 535 this.elem = elem; 536 this.elemCount = elemCount; 537 this.totalCount = elemCount; 538 this.distinctElements = 1; 539 this.height = 1; 540 this.left = null; 541 this.right = null; 542 } 543 544 public int count(Comparator<? super E> comparator, E e) { 545 int cmp = comparator.compare(e, elem); 546 if (cmp < 0) { 547 return (left == null) ? 0 : left.count(comparator, e); 548 } else if (cmp > 0) { 549 return (right == null) ? 0 : right.count(comparator, e); 550 } else { 551 return elemCount; 552 } 553 } 554 555 private AvlNode<E> addRightChild(E e, int count) { 556 right = new AvlNode<E>(e, count); 557 successor(this, right, succ); 558 height = Math.max(2, height); 559 distinctElements++; 560 totalCount += count; 561 return this; 562 } 563 564 private AvlNode<E> addLeftChild(E e, int count) { 565 left = new AvlNode<E>(e, count); 566 successor(pred, left, this); 567 height = Math.max(2, height); 568 distinctElements++; 569 totalCount += count; 570 return this; 571 } 572 573 AvlNode<E> add(Comparator<? super E> comparator, @Nullable E e, int count, int[] result) { 574 /* 575 * It speeds things up considerably to unconditionally add count to totalCount here, 576 * but that destroys failure atomicity in the case of count overflow. =( 577 */ 578 int cmp = comparator.compare(e, elem); 579 if (cmp < 0) { 580 AvlNode<E> initLeft = left; 581 if (initLeft == null) { 582 result[0] = 0; 583 return addLeftChild(e, count); 584 } 585 int initHeight = initLeft.height; 586 587 left = initLeft.add(comparator, e, count, result); 588 if (result[0] == 0) { 589 distinctElements++; 590 } 591 this.totalCount += count; 592 return (left.height == initHeight) ? this : rebalance(); 593 } else if (cmp > 0) { 594 AvlNode<E> initRight = right; 595 if (initRight == null) { 596 result[0] = 0; 597 return addRightChild(e, count); 598 } 599 int initHeight = initRight.height; 600 601 right = initRight.add(comparator, e, count, result); 602 if (result[0] == 0) { 603 distinctElements++; 604 } 605 this.totalCount += count; 606 return (right.height == initHeight) ? this : rebalance(); 607 } 608 609 // adding count to me! No rebalance possible. 610 result[0] = elemCount; 611 long resultCount = (long) elemCount + count; 612 checkArgument(resultCount <= Integer.MAX_VALUE); 613 this.elemCount += count; 614 this.totalCount += count; 615 return this; 616 } 617 618 AvlNode<E> remove(Comparator<? super E> comparator, @Nullable E e, int count, int[] result) { 619 int cmp = comparator.compare(e, elem); 620 if (cmp < 0) { 621 AvlNode<E> initLeft = left; 622 if (initLeft == null) { 623 result[0] = 0; 624 return this; 625 } 626 627 left = initLeft.remove(comparator, e, count, result); 628 629 if (result[0] > 0) { 630 if (count >= result[0]) { 631 this.distinctElements--; 632 this.totalCount -= result[0]; 633 } else { 634 this.totalCount -= count; 635 } 636 } 637 return (result[0] == 0) ? this : rebalance(); 638 } else if (cmp > 0) { 639 AvlNode<E> initRight = right; 640 if (initRight == null) { 641 result[0] = 0; 642 return this; 643 } 644 645 right = initRight.remove(comparator, e, count, result); 646 647 if (result[0] > 0) { 648 if (count >= result[0]) { 649 this.distinctElements--; 650 this.totalCount -= result[0]; 651 } else { 652 this.totalCount -= count; 653 } 654 } 655 return rebalance(); 656 } 657 658 // removing count from me! 659 result[0] = elemCount; 660 if (count >= elemCount) { 661 return deleteMe(); 662 } else { 663 this.elemCount -= count; 664 this.totalCount -= count; 665 return this; 666 } 667 } 668 669 AvlNode<E> setCount(Comparator<? super E> comparator, @Nullable E e, int count, int[] result) { 670 int cmp = comparator.compare(e, elem); 671 if (cmp < 0) { 672 AvlNode<E> initLeft = left; 673 if (initLeft == null) { 674 result[0] = 0; 675 return (count > 0) ? addLeftChild(e, count) : this; 676 } 677 678 left = initLeft.setCount(comparator, e, count, result); 679 680 if (count == 0 && result[0] != 0) { 681 this.distinctElements--; 682 } else if (count > 0 && result[0] == 0) { 683 this.distinctElements++; 684 } 685 686 this.totalCount += count - result[0]; 687 return rebalance(); 688 } else if (cmp > 0) { 689 AvlNode<E> initRight = right; 690 if (initRight == null) { 691 result[0] = 0; 692 return (count > 0) ? addRightChild(e, count) : this; 693 } 694 695 right = initRight.setCount(comparator, e, count, result); 696 697 if (count == 0 && result[0] != 0) { 698 this.distinctElements--; 699 } else if (count > 0 && result[0] == 0) { 700 this.distinctElements++; 701 } 702 703 this.totalCount += count - result[0]; 704 return rebalance(); 705 } 706 707 // setting my count 708 result[0] = elemCount; 709 if (count == 0) { 710 return deleteMe(); 711 } 712 this.totalCount += count - elemCount; 713 this.elemCount = count; 714 return this; 715 } 716 717 AvlNode<E> setCount( 718 Comparator<? super E> comparator, 719 @Nullable E e, 720 int expectedCount, 721 int newCount, 722 int[] result) { 723 int cmp = comparator.compare(e, elem); 724 if (cmp < 0) { 725 AvlNode<E> initLeft = left; 726 if (initLeft == null) { 727 result[0] = 0; 728 if (expectedCount == 0 && newCount > 0) { 729 return addLeftChild(e, newCount); 730 } 731 return this; 732 } 733 734 left = initLeft.setCount(comparator, e, expectedCount, newCount, result); 735 736 if (result[0] == expectedCount) { 737 if (newCount == 0 && result[0] != 0) { 738 this.distinctElements--; 739 } else if (newCount > 0 && result[0] == 0) { 740 this.distinctElements++; 741 } 742 this.totalCount += newCount - result[0]; 743 } 744 return rebalance(); 745 } else if (cmp > 0) { 746 AvlNode<E> initRight = right; 747 if (initRight == null) { 748 result[0] = 0; 749 if (expectedCount == 0 && newCount > 0) { 750 return addRightChild(e, newCount); 751 } 752 return this; 753 } 754 755 right = initRight.setCount(comparator, e, expectedCount, newCount, result); 756 757 if (result[0] == expectedCount) { 758 if (newCount == 0 && result[0] != 0) { 759 this.distinctElements--; 760 } else if (newCount > 0 && result[0] == 0) { 761 this.distinctElements++; 762 } 763 this.totalCount += newCount - result[0]; 764 } 765 return rebalance(); 766 } 767 768 // setting my count 769 result[0] = elemCount; 770 if (expectedCount == elemCount) { 771 if (newCount == 0) { 772 return deleteMe(); 773 } 774 this.totalCount += newCount - elemCount; 775 this.elemCount = newCount; 776 } 777 return this; 778 } 779 780 private AvlNode<E> deleteMe() { 781 int oldElemCount = this.elemCount; 782 this.elemCount = 0; 783 successor(pred, succ); 784 if (left == null) { 785 return right; 786 } else if (right == null) { 787 return left; 788 } else if (left.height >= right.height) { 789 AvlNode<E> newTop = pred; 790 // newTop is the maximum node in my left subtree 791 newTop.left = left.removeMax(newTop); 792 newTop.right = right; 793 newTop.distinctElements = distinctElements - 1; 794 newTop.totalCount = totalCount - oldElemCount; 795 return newTop.rebalance(); 796 } else { 797 AvlNode<E> newTop = succ; 798 newTop.right = right.removeMin(newTop); 799 newTop.left = left; 800 newTop.distinctElements = distinctElements - 1; 801 newTop.totalCount = totalCount - oldElemCount; 802 return newTop.rebalance(); 803 } 804 } 805 806 // Removes the minimum node from this subtree to be reused elsewhere 807 private AvlNode<E> removeMin(AvlNode<E> node) { 808 if (left == null) { 809 return right; 810 } else { 811 left = left.removeMin(node); 812 distinctElements--; 813 totalCount -= node.elemCount; 814 return rebalance(); 815 } 816 } 817 818 // Removes the maximum node from this subtree to be reused elsewhere 819 private AvlNode<E> removeMax(AvlNode<E> node) { 820 if (right == null) { 821 return left; 822 } else { 823 right = right.removeMax(node); 824 distinctElements--; 825 totalCount -= node.elemCount; 826 return rebalance(); 827 } 828 } 829 830 private void recomputeMultiset() { 831 this.distinctElements = 1 + TreeMultiset.distinctElements(left) 832 + TreeMultiset.distinctElements(right); 833 this.totalCount = elemCount + totalCount(left) + totalCount(right); 834 } 835 836 private void recomputeHeight() { 837 this.height = 1 + Math.max(height(left), height(right)); 838 } 839 840 private void recompute() { 841 recomputeMultiset(); 842 recomputeHeight(); 843 } 844 845 private AvlNode<E> rebalance() { 846 switch (balanceFactor()) { 847 case -2: 848 if (right.balanceFactor() > 0) { 849 right = right.rotateRight(); 850 } 851 return rotateLeft(); 852 case 2: 853 if (left.balanceFactor() < 0) { 854 left = left.rotateLeft(); 855 } 856 return rotateRight(); 857 default: 858 recomputeHeight(); 859 return this; 860 } 861 } 862 863 private int balanceFactor() { 864 return height(left) - height(right); 865 } 866 867 private AvlNode<E> rotateLeft() { 868 checkState(right != null); 869 AvlNode<E> newTop = right; 870 this.right = newTop.left; 871 newTop.left = this; 872 newTop.totalCount = this.totalCount; 873 newTop.distinctElements = this.distinctElements; 874 this.recompute(); 875 newTop.recomputeHeight(); 876 return newTop; 877 } 878 879 private AvlNode<E> rotateRight() { 880 checkState(left != null); 881 AvlNode<E> newTop = left; 882 this.left = newTop.right; 883 newTop.right = this; 884 newTop.totalCount = this.totalCount; 885 newTop.distinctElements = this.distinctElements; 886 this.recompute(); 887 newTop.recomputeHeight(); 888 return newTop; 889 } 890 891 private static long totalCount(@Nullable AvlNode<?> node) { 892 return (node == null) ? 0 : node.totalCount; 893 } 894 895 private static int height(@Nullable AvlNode<?> node) { 896 return (node == null) ? 0 : node.height; 897 } 898 899 @Nullable private AvlNode<E> ceiling(Comparator<? super E> comparator, E e) { 900 int cmp = comparator.compare(e, elem); 901 if (cmp < 0) { 902 return (left == null) ? this : MoreObjects.firstNonNull(left.ceiling(comparator, e), this); 903 } else if (cmp == 0) { 904 return this; 905 } else { 906 return (right == null) ? null : right.ceiling(comparator, e); 907 } 908 } 909 910 @Nullable private AvlNode<E> floor(Comparator<? super E> comparator, E e) { 911 int cmp = comparator.compare(e, elem); 912 if (cmp > 0) { 913 return (right == null) ? this : MoreObjects.firstNonNull(right.floor(comparator, e), this); 914 } else if (cmp == 0) { 915 return this; 916 } else { 917 return (left == null) ? null : left.floor(comparator, e); 918 } 919 } 920 921 @Override 922 public E getElement() { 923 return elem; 924 } 925 926 @Override 927 public int getCount() { 928 return elemCount; 929 } 930 931 @Override 932 public String toString() { 933 return Multisets.immutableEntry(getElement(), getCount()).toString(); 934 } 935 } 936 937 private static <T> void successor(AvlNode<T> a, AvlNode<T> b) { 938 a.succ = b; 939 b.pred = a; 940 } 941 942 private static <T> void successor(AvlNode<T> a, AvlNode<T> b, AvlNode<T> c) { 943 successor(a, b); 944 successor(b, c); 945 } 946 947 /* 948 * TODO(jlevy): Decide whether entrySet() should return entries with an equals() method that 949 * calls the comparator to compare the two keys. If that change is made, 950 * AbstractMultiset.equals() can simply check whether two multisets have equal entry sets. 951 */ 952 953 /** 954 * @serialData the comparator, the number of distinct elements, the first element, its count, the 955 * second element, its count, and so on 956 */ 957 @GwtIncompatible("java.io.ObjectOutputStream") 958 private void writeObject(ObjectOutputStream stream) throws IOException { 959 stream.defaultWriteObject(); 960 stream.writeObject(elementSet().comparator()); 961 Serialization.writeMultiset(this, stream); 962 } 963 964 @GwtIncompatible("java.io.ObjectInputStream") 965 private void readObject(ObjectInputStream stream) throws IOException, ClassNotFoundException { 966 stream.defaultReadObject(); 967 @SuppressWarnings("unchecked") 968 // reading data stored by writeObject 969 Comparator<? super E> comparator = (Comparator<? super E>) stream.readObject(); 970 Serialization.getFieldSetter(AbstractSortedMultiset.class, "comparator").set(this, comparator); 971 Serialization.getFieldSetter(TreeMultiset.class, "range").set( 972 this, 973 GeneralRange.all(comparator)); 974 Serialization.getFieldSetter(TreeMultiset.class, "rootReference").set( 975 this, 976 new Reference<AvlNode<E>>()); 977 AvlNode<E> header = new AvlNode<E>(null, 1); 978 Serialization.getFieldSetter(TreeMultiset.class, "header").set(this, header); 979 successor(header, header); 980 Serialization.populateMultiset(this, stream); 981 } 982 983 @GwtIncompatible("not needed in emulated source") private static final long serialVersionUID = 1; 984 } 985