github.com/keysonZZZ/kmg@v0.0.0-20151121023212-05317bfd7d39/kmgRpc/kmgRpcJava/java/src/com/google/gson/internal/LinkedTreeMap.java (about)

     1  /*
     2   * Copyright (C) 2010 The Android Open Source Project
     3   * Copyright (C) 2012 Google Inc.
     4   *
     5   * Licensed under the Apache License, Version 2.0 (the "License");
     6   * you may not use this file except in compliance with the License.
     7   * You may obtain a copy of the License at
     8   *
     9   *      http://www.apache.org/licenses/LICENSE-2.0
    10   *
    11   * Unless required by applicable law or agreed to in writing, software
    12   * distributed under the License is distributed on an "AS IS" BASIS,
    13   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    14   * See the License for the specific language governing permissions and
    15   * limitations under the License.
    16   */
    17  
    18  package com.google.gson.internal;
    19  
    20  import java.io.ObjectStreamException;
    21  import java.io.Serializable;
    22  import java.util.AbstractMap;
    23  import java.util.AbstractSet;
    24  import java.util.Comparator;
    25  import java.util.ConcurrentModificationException;
    26  import java.util.Iterator;
    27  import java.util.LinkedHashMap;
    28  import java.util.NoSuchElementException;
    29  import java.util.Set;
    30  
    31  /**
    32   * A map of comparable keys to values. Unlike {@code TreeMap}, this class uses
    33   * insertion order for iteration order. Comparison order is only used as an
    34   * optimization for efficient insertion and removal.
    35   *
    36   * <p>This implementation was derived from Android 4.1's TreeMap class.
    37   */
    38  public final class LinkedTreeMap<K, V> extends AbstractMap<K, V> implements Serializable {
    39    @SuppressWarnings({ "unchecked", "rawtypes" }) // to avoid Comparable<Comparable<Comparable<...>>>
    40    private static final Comparator<Comparable> NATURAL_ORDER = new Comparator<Comparable>() {
    41      public int compare(Comparable a, Comparable b) {
    42        return a.compareTo(b);
    43      }
    44    };
    45  
    46    Comparator<? super K> comparator;
    47    Node<K, V> root;
    48    int size = 0;
    49    int modCount = 0;
    50  
    51    // Used to preserve iteration order
    52    final Node<K, V> header = new Node<K, V>();
    53  
    54    /**
    55     * Create a natural order, empty tree map whose keys must be mutually
    56     * comparable and non-null.
    57     */
    58    @SuppressWarnings("unchecked") // unsafe! this assumes K is comparable
    59    public LinkedTreeMap() {
    60      this((Comparator<? super K>) NATURAL_ORDER);
    61    }
    62  
    63    /**
    64     * Create a tree map ordered by {@code comparator}. This map's keys may only
    65     * be null if {@code comparator} permits.
    66     *
    67     * @param comparator the comparator to order elements with, or {@code null} to
    68     *     use the natural ordering.
    69     */
    70    @SuppressWarnings({ "unchecked", "rawtypes" }) // unsafe! if comparator is null, this assumes K is comparable
    71    public LinkedTreeMap(Comparator<? super K> comparator) {
    72      this.comparator = comparator != null
    73          ? comparator
    74          : (Comparator) NATURAL_ORDER;
    75    }
    76  
    77    @Override public int size() {
    78      return size;
    79    }
    80  
    81    @Override public V get(Object key) {
    82      Node<K, V> node = findByObject(key);
    83      return node != null ? node.value : null;
    84    }
    85  
    86    @Override public boolean containsKey(Object key) {
    87      return findByObject(key) != null;
    88    }
    89  
    90    @Override public V put(K key, V value) {
    91      if (key == null) {
    92        throw new NullPointerException("key == null");
    93      }
    94      Node<K, V> created = find(key, true);
    95      V result = created.value;
    96      created.value = value;
    97      return result;
    98    }
    99  
   100    @Override public void clear() {
   101      root = null;
   102      size = 0;
   103      modCount++;
   104  
   105      // Clear iteration order
   106      Node<K, V> header = this.header;
   107      header.next = header.prev = header;
   108    }
   109  
   110    @Override public V remove(Object key) {
   111      Node<K, V> node = removeInternalByKey(key);
   112      return node != null ? node.value : null;
   113    }
   114  
   115    /**
   116     * Returns the node at or adjacent to the given key, creating it if requested.
   117     *
   118     * @throws ClassCastException if {@code key} and the tree's keys aren't
   119     *     mutually comparable.
   120     */
   121    Node<K, V> find(K key, boolean create) {
   122      Comparator<? super K> comparator = this.comparator;
   123      Node<K, V> nearest = root;
   124      int comparison = 0;
   125  
   126      if (nearest != null) {
   127        // Micro-optimization: avoid polymorphic calls to Comparator.compare().
   128        @SuppressWarnings("unchecked") // Throws a ClassCastException below if there's trouble.
   129            Comparable<Object> comparableKey = (comparator == NATURAL_ORDER)
   130            ? (Comparable<Object>) key
   131            : null;
   132  
   133        while (true) {
   134          comparison = (comparableKey != null)
   135              ? comparableKey.compareTo(nearest.key)
   136              : comparator.compare(key, nearest.key);
   137  
   138          // We found the requested key.
   139          if (comparison == 0) {
   140            return nearest;
   141          }
   142  
   143          // If it exists, the key is in a subtree. Go deeper.
   144          Node<K, V> child = (comparison < 0) ? nearest.left : nearest.right;
   145          if (child == null) {
   146            break;
   147          }
   148  
   149          nearest = child;
   150        }
   151      }
   152  
   153      // The key doesn't exist in this tree.
   154      if (!create) {
   155        return null;
   156      }
   157  
   158      // Create the node and add it to the tree or the table.
   159      Node<K, V> header = this.header;
   160      Node<K, V> created;
   161      if (nearest == null) {
   162        // Check that the value is comparable if we didn't do any comparisons.
   163        if (comparator == NATURAL_ORDER && !(key instanceof Comparable)) {
   164          throw new ClassCastException(key.getClass().getName() + " is not Comparable");
   165        }
   166        created = new Node<K, V>(nearest, key, header, header.prev);
   167        root = created;
   168      } else {
   169        created = new Node<K, V>(nearest, key, header, header.prev);
   170        if (comparison < 0) { // nearest.key is higher
   171          nearest.left = created;
   172        } else { // comparison > 0, nearest.key is lower
   173          nearest.right = created;
   174        }
   175        rebalance(nearest, true);
   176      }
   177      size++;
   178      modCount++;
   179  
   180      return created;
   181    }
   182  
   183    @SuppressWarnings("unchecked")
   184    Node<K, V> findByObject(Object key) {
   185      try {
   186        return key != null ? find((K) key, false) : null;
   187      } catch (ClassCastException e) {
   188        return null;
   189      }
   190    }
   191  
   192    /**
   193     * Returns this map's entry that has the same key and value as {@code
   194     * entry}, or null if this map has no such entry.
   195     *
   196     * <p>This method uses the comparator for key equality rather than {@code
   197     * equals}. If this map's comparator isn't consistent with equals (such as
   198     * {@code String.CASE_INSENSITIVE_ORDER}), then {@code remove()} and {@code
   199     * contains()} will violate the collections API.
   200     */
   201    Node<K, V> findByEntry(Entry<?, ?> entry) {
   202      Node<K, V> mine = findByObject(entry.getKey());
   203      boolean valuesEqual = mine != null && equal(mine.value, entry.getValue());
   204      return valuesEqual ? mine : null;
   205    }
   206  
   207    private boolean equal(Object a, Object b) {
   208      return a == b || (a != null && a.equals(b));
   209    }
   210  
   211    /**
   212     * Removes {@code node} from this tree, rearranging the tree's structure as
   213     * necessary.
   214     *
   215     * @param unlink true to also unlink this node from the iteration linked list.
   216     */
   217    void removeInternal(Node<K, V> node, boolean unlink) {
   218      if (unlink) {
   219        node.prev.next = node.next;
   220        node.next.prev = node.prev;
   221      }
   222  
   223      Node<K, V> left = node.left;
   224      Node<K, V> right = node.right;
   225      Node<K, V> originalParent = node.parent;
   226      if (left != null && right != null) {
   227  
   228        /*
   229         * To remove a node with both left and right subtrees, move an
   230         * adjacent node from one of those subtrees into this node's place.
   231         *
   232         * Removing the adjacent node may change this node's subtrees. This
   233         * node may no longer have two subtrees once the adjacent node is
   234         * gone!
   235         */
   236  
   237        Node<K, V> adjacent = (left.height > right.height) ? left.last() : right.first();
   238        removeInternal(adjacent, false); // takes care of rebalance and size--
   239  
   240        int leftHeight = 0;
   241        left = node.left;
   242        if (left != null) {
   243          leftHeight = left.height;
   244          adjacent.left = left;
   245          left.parent = adjacent;
   246          node.left = null;
   247        }
   248  
   249        int rightHeight = 0;
   250        right = node.right;
   251        if (right != null) {
   252          rightHeight = right.height;
   253          adjacent.right = right;
   254          right.parent = adjacent;
   255          node.right = null;
   256        }
   257  
   258        adjacent.height = Math.max(leftHeight, rightHeight) + 1;
   259        replaceInParent(node, adjacent);
   260        return;
   261      } else if (left != null) {
   262        replaceInParent(node, left);
   263        node.left = null;
   264      } else if (right != null) {
   265        replaceInParent(node, right);
   266        node.right = null;
   267      } else {
   268        replaceInParent(node, null);
   269      }
   270  
   271      rebalance(originalParent, false);
   272      size--;
   273      modCount++;
   274    }
   275  
   276    Node<K, V> removeInternalByKey(Object key) {
   277      Node<K, V> node = findByObject(key);
   278      if (node != null) {
   279        removeInternal(node, true);
   280      }
   281      return node;
   282    }
   283  
   284    private void replaceInParent(Node<K, V> node, Node<K, V> replacement) {
   285      Node<K, V> parent = node.parent;
   286      node.parent = null;
   287      if (replacement != null) {
   288        replacement.parent = parent;
   289      }
   290  
   291      if (parent != null) {
   292        if (parent.left == node) {
   293          parent.left = replacement;
   294        } else {
   295          assert (parent.right == node);
   296          parent.right = replacement;
   297        }
   298      } else {
   299        root = replacement;
   300      }
   301    }
   302  
   303    /**
   304     * Rebalances the tree by making any AVL rotations necessary between the
   305     * newly-unbalanced node and the tree's root.
   306     *
   307     * @param insert true if the node was unbalanced by an insert; false if it
   308     *     was by a removal.
   309     */
   310    private void rebalance(Node<K, V> unbalanced, boolean insert) {
   311      for (Node<K, V> node = unbalanced; node != null; node = node.parent) {
   312        Node<K, V> left = node.left;
   313        Node<K, V> right = node.right;
   314        int leftHeight = left != null ? left.height : 0;
   315        int rightHeight = right != null ? right.height : 0;
   316  
   317        int delta = leftHeight - rightHeight;
   318        if (delta == -2) {
   319          Node<K, V> rightLeft = right.left;
   320          Node<K, V> rightRight = right.right;
   321          int rightRightHeight = rightRight != null ? rightRight.height : 0;
   322          int rightLeftHeight = rightLeft != null ? rightLeft.height : 0;
   323  
   324          int rightDelta = rightLeftHeight - rightRightHeight;
   325          if (rightDelta == -1 || (rightDelta == 0 && !insert)) {
   326            rotateLeft(node); // AVL right right
   327          } else {
   328            assert (rightDelta == 1);
   329            rotateRight(right); // AVL right left
   330            rotateLeft(node);
   331          }
   332          if (insert) {
   333            break; // no further rotations will be necessary
   334          }
   335  
   336        } else if (delta == 2) {
   337          Node<K, V> leftLeft = left.left;
   338          Node<K, V> leftRight = left.right;
   339          int leftRightHeight = leftRight != null ? leftRight.height : 0;
   340          int leftLeftHeight = leftLeft != null ? leftLeft.height : 0;
   341  
   342          int leftDelta = leftLeftHeight - leftRightHeight;
   343          if (leftDelta == 1 || (leftDelta == 0 && !insert)) {
   344            rotateRight(node); // AVL left left
   345          } else {
   346            assert (leftDelta == -1);
   347            rotateLeft(left); // AVL left right
   348            rotateRight(node);
   349          }
   350          if (insert) {
   351            break; // no further rotations will be necessary
   352          }
   353  
   354        } else if (delta == 0) {
   355          node.height = leftHeight + 1; // leftHeight == rightHeight
   356          if (insert) {
   357            break; // the insert caused balance, so rebalancing is done!
   358          }
   359  
   360        } else {
   361          assert (delta == -1 || delta == 1);
   362          node.height = Math.max(leftHeight, rightHeight) + 1;
   363          if (!insert) {
   364            break; // the height hasn't changed, so rebalancing is done!
   365          }
   366        }
   367      }
   368    }
   369  
   370    /**
   371     * Rotates the subtree so that its root's right child is the new root.
   372     */
   373    private void rotateLeft(Node<K, V> root) {
   374      Node<K, V> left = root.left;
   375      Node<K, V> pivot = root.right;
   376      Node<K, V> pivotLeft = pivot.left;
   377      Node<K, V> pivotRight = pivot.right;
   378  
   379      // move the pivot's left child to the root's right
   380      root.right = pivotLeft;
   381      if (pivotLeft != null) {
   382        pivotLeft.parent = root;
   383      }
   384  
   385      replaceInParent(root, pivot);
   386  
   387      // move the root to the pivot's left
   388      pivot.left = root;
   389      root.parent = pivot;
   390  
   391      // fix heights
   392      root.height = Math.max(left != null ? left.height : 0,
   393          pivotLeft != null ? pivotLeft.height : 0) + 1;
   394      pivot.height = Math.max(root.height,
   395          pivotRight != null ? pivotRight.height : 0) + 1;
   396    }
   397  
   398    /**
   399     * Rotates the subtree so that its root's left child is the new root.
   400     */
   401    private void rotateRight(Node<K, V> root) {
   402      Node<K, V> pivot = root.left;
   403      Node<K, V> right = root.right;
   404      Node<K, V> pivotLeft = pivot.left;
   405      Node<K, V> pivotRight = pivot.right;
   406  
   407      // move the pivot's right child to the root's left
   408      root.left = pivotRight;
   409      if (pivotRight != null) {
   410        pivotRight.parent = root;
   411      }
   412  
   413      replaceInParent(root, pivot);
   414  
   415      // move the root to the pivot's right
   416      pivot.right = root;
   417      root.parent = pivot;
   418  
   419      // fixup heights
   420      root.height = Math.max(right != null ? right.height : 0,
   421          pivotRight != null ? pivotRight.height : 0) + 1;
   422      pivot.height = Math.max(root.height,
   423          pivotLeft != null ? pivotLeft.height : 0) + 1;
   424    }
   425  
   426    private EntrySet entrySet;
   427    private KeySet keySet;
   428  
   429    @Override public Set<Entry<K, V>> entrySet() {
   430      EntrySet result = entrySet;
   431      return result != null ? result : (entrySet = new EntrySet());
   432    }
   433  
   434    @Override public Set<K> keySet() {
   435      KeySet result = keySet;
   436      return result != null ? result : (keySet = new KeySet());
   437    }
   438  
   439    static final class Node<K, V> implements Entry<K, V> {
   440      Node<K, V> parent;
   441      Node<K, V> left;
   442      Node<K, V> right;
   443      Node<K, V> next;
   444      Node<K, V> prev;
   445      final K key;
   446      V value;
   447      int height;
   448  
   449      /** Create the header entry */
   450      Node() {
   451        key = null;
   452        next = prev = this;
   453      }
   454  
   455      /** Create a regular entry */
   456      Node(Node<K, V> parent, K key, Node<K, V> next, Node<K, V> prev) {
   457        this.parent = parent;
   458        this.key = key;
   459        this.height = 1;
   460        this.next = next;
   461        this.prev = prev;
   462        prev.next = this;
   463        next.prev = this;
   464      }
   465  
   466      public K getKey() {
   467        return key;
   468      }
   469  
   470      public V getValue() {
   471        return value;
   472      }
   473  
   474      public V setValue(V value) {
   475        V oldValue = this.value;
   476        this.value = value;
   477        return oldValue;
   478      }
   479  
   480      @SuppressWarnings("rawtypes")
   481      @Override public boolean equals(Object o) {
   482        if (o instanceof Entry) {
   483          Entry other = (Entry) o;
   484          return (key == null ? other.getKey() == null : key.equals(other.getKey()))
   485              && (value == null ? other.getValue() == null : value.equals(other.getValue()));
   486        }
   487        return false;
   488      }
   489  
   490      @Override public int hashCode() {
   491        return (key == null ? 0 : key.hashCode())
   492            ^ (value == null ? 0 : value.hashCode());
   493      }
   494  
   495      @Override public String toString() {
   496        return key + "=" + value;
   497      }
   498  
   499      /**
   500       * Returns the first node in this subtree.
   501       */
   502      public Node<K, V> first() {
   503        Node<K, V> node = this;
   504        Node<K, V> child = node.left;
   505        while (child != null) {
   506          node = child;
   507          child = node.left;
   508        }
   509        return node;
   510      }
   511  
   512      /**
   513       * Returns the last node in this subtree.
   514       */
   515      public Node<K, V> last() {
   516        Node<K, V> node = this;
   517        Node<K, V> child = node.right;
   518        while (child != null) {
   519          node = child;
   520          child = node.right;
   521        }
   522        return node;
   523      }
   524    }
   525  
   526    private abstract class LinkedTreeMapIterator<T> implements Iterator<T> {
   527      Node<K, V> next = header.next;
   528      Node<K, V> lastReturned = null;
   529      int expectedModCount = modCount;
   530  
   531      public final boolean hasNext() {
   532        return next != header;
   533      }
   534  
   535      final Node<K, V> nextNode() {
   536        Node<K, V> e = next;
   537        if (e == header) {
   538          throw new NoSuchElementException();
   539        }
   540        if (modCount != expectedModCount) {
   541          throw new ConcurrentModificationException();
   542        }
   543        next = e.next;
   544        return lastReturned = e;
   545      }
   546  
   547      public final void remove() {
   548        if (lastReturned == null) {
   549          throw new IllegalStateException();
   550        }
   551        removeInternal(lastReturned, true);
   552        lastReturned = null;
   553        expectedModCount = modCount;
   554      }
   555    }
   556  
   557    class EntrySet extends AbstractSet<Entry<K, V>> {
   558      @Override public int size() {
   559        return size;
   560      }
   561  
   562      @Override public Iterator<Entry<K, V>> iterator() {
   563        return new LinkedTreeMapIterator<Entry<K, V>>() {
   564          public Entry<K, V> next() {
   565            return nextNode();
   566          }
   567        };
   568      }
   569  
   570      @Override public boolean contains(Object o) {
   571        return o instanceof Entry && findByEntry((Entry<?, ?>) o) != null;
   572      }
   573  
   574      @Override public boolean remove(Object o) {
   575        if (!(o instanceof Entry)) {
   576          return false;
   577        }
   578  
   579        Node<K, V> node = findByEntry((Entry<?, ?>) o);
   580        if (node == null) {
   581          return false;
   582        }
   583        removeInternal(node, true);
   584        return true;
   585      }
   586  
   587      @Override public void clear() {
   588        LinkedTreeMap.this.clear();
   589      }
   590    }
   591  
   592    final class KeySet extends AbstractSet<K> {
   593      @Override public int size() {
   594        return size;
   595      }
   596  
   597      @Override public Iterator<K> iterator() {
   598        return new LinkedTreeMapIterator<K>() {
   599          public K next() {
   600            return nextNode().key;
   601          }
   602        };
   603      }
   604  
   605      @Override public boolean contains(Object o) {
   606        return containsKey(o);
   607      }
   608  
   609      @Override public boolean remove(Object key) {
   610        return removeInternalByKey(key) != null;
   611      }
   612  
   613      @Override public void clear() {
   614        LinkedTreeMap.this.clear();
   615      }
   616    }
   617  
   618    /**
   619     * If somebody is unlucky enough to have to serialize one of these, serialize
   620     * it as a LinkedHashMap so that they won't need Gson on the other side to
   621     * deserialize it. Using serialization defeats our DoS defence, so most apps
   622     * shouldn't use it.
   623     */
   624    private Object writeReplace() throws ObjectStreamException {
   625      return new LinkedHashMap<K, V>(this);
   626    }
   627  }