mirror of
https://github.com/Eaglercraft-TeaVM-Fork/eagler-teavm.git
synced 2025-01-09 00:14:10 -08:00
Rewritten TreeMap passes most of tests
This commit is contained in:
parent
4be4c4a843
commit
161c4c76b7
|
@ -16,11 +16,9 @@
|
||||||
package org.teavm.classlib.java.util;
|
package org.teavm.classlib.java.util;
|
||||||
|
|
||||||
import org.teavm.classlib.java.io.TSerializable;
|
import org.teavm.classlib.java.io.TSerializable;
|
||||||
import org.teavm.classlib.java.lang.ObjectNativeGenerator;
|
|
||||||
import org.teavm.classlib.java.lang.TCloneNotSupportedException;
|
import org.teavm.classlib.java.lang.TCloneNotSupportedException;
|
||||||
import org.teavm.classlib.java.lang.TObject;
|
import org.teavm.classlib.java.lang.TObject;
|
||||||
import org.teavm.classlib.java.lang.TUnsupportedOperationException;
|
import org.teavm.classlib.java.lang.TUnsupportedOperationException;
|
||||||
import org.teavm.dependency.PluggableDependency;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
@ -266,7 +264,6 @@ public abstract class TAbstractMap<K, V> extends TObject implements TMap<K, V> {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@PluggableDependency(ObjectNativeGenerator.class)
|
|
||||||
protected Object clone() throws TCloneNotSupportedException {
|
protected Object clone() throws TCloneNotSupportedException {
|
||||||
TAbstractMap<?, ?> copy = (TAbstractMap<?, ?>)super.clone();
|
TAbstractMap<?, ?> copy = (TAbstractMap<?, ?>)super.clone();
|
||||||
copy.cachedKeySet = null;
|
copy.cachedKeySet = null;
|
||||||
|
|
|
@ -13,58 +13,23 @@
|
||||||
* See the License for the specific language governing permissions and
|
* See the License for the specific language governing permissions and
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
/*
|
|
||||||
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
||||||
* contributor license agreements. See the NOTICE file distributed with
|
|
||||||
* this work for additional information regarding copyright ownership.
|
|
||||||
* The ASF licenses this file to You 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 org.teavm.classlib.java.util;
|
package org.teavm.classlib.java.util;
|
||||||
|
|
||||||
import org.teavm.classlib.java.io.TSerializable;
|
import org.teavm.classlib.java.io.TSerializable;
|
||||||
import org.teavm.classlib.java.lang.*;
|
import org.teavm.classlib.java.lang.*;
|
||||||
|
|
||||||
public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TSerializable, TSortedMap<K, V> {
|
public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TSerializable, TSortedMap<K, V> {
|
||||||
class TreeNode implements Entry<K, V> {
|
static class TreeNode<K, V> extends SimpleEntry<K, V> {
|
||||||
K key;
|
TreeNode<K, V> left;
|
||||||
V value;
|
TreeNode<K, V> right;
|
||||||
TreeNode left;
|
|
||||||
TreeNode right;
|
|
||||||
int height = 1;
|
int height = 1;
|
||||||
int size = 1;
|
int size = 1;
|
||||||
|
|
||||||
public TreeNode(K key) {
|
public TreeNode(K key) {
|
||||||
this.key = key;
|
super(key, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
public TreeNode<K, V> balance() {
|
||||||
public K getKey() {
|
|
||||||
return key;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public V getValue() {
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public V setValue(V value) {
|
|
||||||
V old = this.value;
|
|
||||||
this.value = value;
|
|
||||||
return old;
|
|
||||||
}
|
|
||||||
|
|
||||||
public TreeNode balance() {
|
|
||||||
int factor = factor();
|
int factor = factor();
|
||||||
if (factor == 2) {
|
if (factor == 2) {
|
||||||
if (right.factor() < 0) {
|
if (right.factor() < 0) {
|
||||||
|
@ -85,8 +50,8 @@ public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TS
|
||||||
return (right != null ? right.height : 0) - (left != null ? left.height : 0);
|
return (right != null ? right.height : 0) - (left != null ? left.height : 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
public TreeNode rotateRight() {
|
public TreeNode<K, V> rotateRight() {
|
||||||
TreeNode left = this.left;
|
TreeNode<K, V> left = this.left;
|
||||||
this.left = left.right;
|
this.left = left.right;
|
||||||
left.right = this;
|
left.right = this;
|
||||||
fix();
|
fix();
|
||||||
|
@ -94,8 +59,8 @@ public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TS
|
||||||
return left;
|
return left;
|
||||||
}
|
}
|
||||||
|
|
||||||
public TreeNode rotateLeft() {
|
public TreeNode<K, V> rotateLeft() {
|
||||||
TreeNode right = this.right;
|
TreeNode<K, V> right = this.right;
|
||||||
this.right = right.left;
|
this.right = right.left;
|
||||||
right.left = this;
|
right.left = this;
|
||||||
fix();
|
fix();
|
||||||
|
@ -115,10 +80,15 @@ public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TS
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
TreeNode root;
|
TreeNode<K, V> root;
|
||||||
private TComparator<? super K> comparator;
|
private TComparator<? super K> comparator;
|
||||||
private TComparator<? super K> originalComparator;
|
private TComparator<? super K> originalComparator;
|
||||||
private int modCount = 0;
|
private int modCount = 0;
|
||||||
|
private EntrySet<K, V> cachedEntrySet;
|
||||||
|
|
||||||
|
public TTreeMap() {
|
||||||
|
this((TComparator<? super K>)null);
|
||||||
|
}
|
||||||
|
|
||||||
public TTreeMap(TComparator<? super K> comparator) {
|
public TTreeMap(TComparator<? super K> comparator) {
|
||||||
this.originalComparator = comparator;
|
this.originalComparator = comparator;
|
||||||
|
@ -133,31 +103,70 @@ public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TS
|
||||||
this.comparator = comparator;
|
this.comparator = comparator;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public TTreeMap(TMap<? extends K, ? extends V> m) {
|
||||||
|
this((TComparator<? super K>)null);
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
Entry<K, V>[] entries = (Entry<K, V>[])new Entry<?, ?>[m.size()];
|
||||||
|
entries = m.entrySet().toArray(entries);
|
||||||
|
TArrays.sort(entries, new TComparator<Entry<K, V>>() {
|
||||||
|
@Override public int compare(Entry<K, V> o1, Entry<K, V> o2) {
|
||||||
|
return comparator.compare(o1.getKey(), o2.getKey());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
fillMap(entries);
|
||||||
|
}
|
||||||
|
|
||||||
|
public TTreeMap(TSortedMap<K, ? extends V> m) {
|
||||||
|
this(m.comparator());
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
Entry<K, V>[] entries = (Entry<K, V>[])new Entry<?, ?>[m.size()];
|
||||||
|
entries = m.entrySet().toArray(entries);
|
||||||
|
fillMap(entries);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void fillMap(Entry<? extends K, ? extends V>[] entries) {
|
||||||
|
root = createNode(entries, 0, entries.length - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
private TreeNode<K, V> createNode(Entry<? extends K, ? extends V>[] entries, int l, int u) {
|
||||||
|
if (l > u) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
int mid = (l + u) / 2;
|
||||||
|
Entry<? extends K, ? extends V> entry = entries[mid];
|
||||||
|
TreeNode<K, V> node = new TreeNode<K, V>(entry.getKey());
|
||||||
|
node.setValue(entry.getValue());
|
||||||
|
node.left = createNode(entries, l, mid - 1);
|
||||||
|
node.right = createNode(entries, mid + 1, u);
|
||||||
|
node.fix();
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public V get(Object key) {
|
public V get(Object key) {
|
||||||
TreeNode node = findNode(key);
|
TreeNode<?, V> node = findExact(key);
|
||||||
return node != null ? node.value : null;
|
return node != null ? node.getValue() : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public V put(K key, V value) {
|
public V put(K key, V value) {
|
||||||
root = getOrCreateNode(root, key);
|
root = getOrCreateNode(root, key);
|
||||||
TreeNode node = findNode(key);
|
TreeNode<?, V> node = findExact(key);
|
||||||
V old = node.value;
|
V old = node.setValue(value);
|
||||||
node.value = value;
|
node.setValue(value);
|
||||||
modCount++;
|
modCount++;
|
||||||
return old;
|
return old;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public V remove(Object key) {
|
public V remove(Object key) {
|
||||||
TreeNode node = findNode(key);
|
TreeNode<?, V> node = findExact(key);
|
||||||
if (node == null) {
|
if (node == null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
root = deleteNode(root, key);
|
root = deleteNode(root, key);
|
||||||
modCount++;
|
modCount++;
|
||||||
return node.value;
|
return node.getValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -173,14 +182,14 @@ public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TS
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean containsKey(Object key) {
|
public boolean containsKey(Object key) {
|
||||||
return findNode(key) != null;
|
return findExact(key) != null;
|
||||||
}
|
}
|
||||||
|
|
||||||
TreeNode findNode(Object key) {
|
TreeNode<?, V> findExact(Object key) {
|
||||||
TreeNode node = root;
|
TreeNode<K, V> node = root;
|
||||||
while (node != null) {
|
while (node != null) {
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
int cmp = comparator.compare((K)key, node.key);
|
int cmp = comparator.compare((K)key, node.getKey());
|
||||||
if (cmp == 0) {
|
if (cmp == 0) {
|
||||||
return node;
|
return node;
|
||||||
} else if (cmp < 0) {
|
} else if (cmp < 0) {
|
||||||
|
@ -192,11 +201,136 @@ public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TS
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
private TreeNode getOrCreateNode(TreeNode root, K key) {
|
TreeNode<K, V> findExactOrNext(Object key) {
|
||||||
if (root == null) {
|
TreeNode<K, V> node = root;
|
||||||
return new TreeNode(key);
|
TreeNode<K, V> lastLeftTurn = null;
|
||||||
|
while (node != null) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
int cmp = comparator.compare((K)key, node.getKey());
|
||||||
|
if (cmp == 0) {
|
||||||
|
return node;
|
||||||
|
} else if (cmp < 0) {
|
||||||
|
lastLeftTurn = node;
|
||||||
|
node = node.left;
|
||||||
|
} else {
|
||||||
|
node = node.right;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
int cmp = comparator.compare(key, root.key);
|
return lastLeftTurn;
|
||||||
|
}
|
||||||
|
|
||||||
|
TreeNode<K, V>[] pathToExactOrNext(Object key) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
TreeNode<K, V>[] path = (TreeNode<K, V>[])new TreeNode<?, ?>[height()];
|
||||||
|
int depth = 0;
|
||||||
|
TreeNode<K, V> node = root;
|
||||||
|
while (node != null) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
int cmp = comparator.compare((K)key, node.getKey());
|
||||||
|
if (cmp == 0) {
|
||||||
|
path[depth++] = node;
|
||||||
|
break;
|
||||||
|
} else if (cmp < 0) {
|
||||||
|
path[depth++] = node;
|
||||||
|
node = node.left;
|
||||||
|
} else {
|
||||||
|
node = node.right;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return TArrays.copyOf(path, depth);
|
||||||
|
}
|
||||||
|
|
||||||
|
TreeNode<K, V> findNext(Object key) {
|
||||||
|
TreeNode<K, V> node = root;
|
||||||
|
TreeNode<K, V> lastLeftTurn = null;
|
||||||
|
while (node != null) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
int cmp = comparator.compare((K)key, node.getKey());
|
||||||
|
if (cmp == 0) {
|
||||||
|
break;
|
||||||
|
} else if (cmp < 0) {
|
||||||
|
lastLeftTurn = node;
|
||||||
|
node = node.left;
|
||||||
|
} else {
|
||||||
|
node = node.right;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return lastLeftTurn;
|
||||||
|
}
|
||||||
|
|
||||||
|
TreeNode<K, V>[] pathToNext(Object key) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
TreeNode<K, V>[] path = (TreeNode<K, V>[])new TreeNode<?, ?>[height()];
|
||||||
|
int depth = 0;
|
||||||
|
TreeNode<K, V> node = root;
|
||||||
|
while (node != null) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
int cmp = comparator.compare((K)key, node.getKey());
|
||||||
|
if (cmp == 0) {
|
||||||
|
break;
|
||||||
|
} else if (cmp < 0) {
|
||||||
|
path[depth++] = node;
|
||||||
|
node = node.left;
|
||||||
|
} else {
|
||||||
|
node = node.right;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return TArrays.copyOf(path, depth);
|
||||||
|
}
|
||||||
|
|
||||||
|
TreeNode<K, V> findExactOrPrev(Object key) {
|
||||||
|
TreeNode<K, V> node = root;
|
||||||
|
TreeNode<K, V> lastRightTurn = null;
|
||||||
|
while (node != null) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
int cmp = comparator.compare((K)key, node.getKey());
|
||||||
|
if (cmp == 0) {
|
||||||
|
return node;
|
||||||
|
} else if (cmp > 0) {
|
||||||
|
lastRightTurn = node;
|
||||||
|
node = node.right;
|
||||||
|
} else {
|
||||||
|
node = node.left;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return lastRightTurn;
|
||||||
|
}
|
||||||
|
|
||||||
|
TreeNode<K, V>[] pathToFirst() {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
TreeNode<K, V>[] path = (TreeNode<K, V>[])new TreeNode<?, ?>[height()];
|
||||||
|
int depth = 0;
|
||||||
|
TreeNode<K, V> node = root;
|
||||||
|
while (node != null) {
|
||||||
|
path[depth++] = node;
|
||||||
|
node = node.left;
|
||||||
|
}
|
||||||
|
return TArrays.copyOf(path, depth);
|
||||||
|
}
|
||||||
|
|
||||||
|
TreeNode<K, V> findPrev(Object key) {
|
||||||
|
TreeNode<K, V> node = root;
|
||||||
|
TreeNode<K, V> lastRightTurn = null;
|
||||||
|
while (node != null) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
int cmp = comparator.compare((K)key, node.getKey());
|
||||||
|
if (cmp == 0) {
|
||||||
|
return node;
|
||||||
|
} else if (cmp > 0) {
|
||||||
|
lastRightTurn = node;
|
||||||
|
node = node.right;
|
||||||
|
} else {
|
||||||
|
node = node.left;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return lastRightTurn;
|
||||||
|
}
|
||||||
|
|
||||||
|
private TreeNode<K, V> getOrCreateNode(TreeNode<K, V> root, K key) {
|
||||||
|
if (root == null) {
|
||||||
|
return new TreeNode<>(key);
|
||||||
|
}
|
||||||
|
int cmp = comparator.compare(key, root.getKey());
|
||||||
if (cmp == 0) {
|
if (cmp == 0) {
|
||||||
return root;
|
return root;
|
||||||
} else if (cmp < 0) {
|
} else if (cmp < 0) {
|
||||||
|
@ -204,15 +338,16 @@ public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TS
|
||||||
} else {
|
} else {
|
||||||
root.right = getOrCreateNode(root.right, key);
|
root.right = getOrCreateNode(root.right, key);
|
||||||
}
|
}
|
||||||
|
root.fix();
|
||||||
return root.balance();
|
return root.balance();
|
||||||
}
|
}
|
||||||
|
|
||||||
private TreeNode deleteNode(TreeNode root, Object key) {
|
private TreeNode<K, V> deleteNode(TreeNode<K, V> root, Object key) {
|
||||||
if (root == null) {
|
if (root == null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
int cmp = comparator.compare((K)key, root.key);
|
int cmp = comparator.compare((K)key, root.getKey());
|
||||||
if (cmp < 0) {
|
if (cmp < 0) {
|
||||||
root.left = deleteNode(root.left, key);
|
root.left = deleteNode(root.left, key);
|
||||||
} else if (cmp > 0) {
|
} else if (cmp > 0) {
|
||||||
|
@ -220,19 +355,19 @@ public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TS
|
||||||
} else if (root.right == null) {
|
} else if (root.right == null) {
|
||||||
return root.left;
|
return root.left;
|
||||||
} else {
|
} else {
|
||||||
TreeNode left = root.left;
|
TreeNode<K, V> left = root.left;
|
||||||
TreeNode right = root.right;
|
TreeNode<K, V> right = root.right;
|
||||||
TreeNode min = right;
|
TreeNode<K, V> min = right;
|
||||||
Object[] pathToMin = new Object[right.height];
|
@SuppressWarnings("unchecked")
|
||||||
|
TreeNode<K, V>[] pathToMin = (TreeNode<K, V>[])new TreeNode<?, ?>[right.height];
|
||||||
int minDepth = 0;
|
int minDepth = 0;
|
||||||
while (min.left != null) {
|
while (min.left != null) {
|
||||||
pathToMin[minDepth++] = min;
|
pathToMin[minDepth++] = min;
|
||||||
min = min.left;
|
min = min.left;
|
||||||
}
|
}
|
||||||
right = min.right;
|
right = min.right;
|
||||||
while (minDepth >= 0) {
|
while (minDepth > 0) {
|
||||||
@SuppressWarnings("unchecked")
|
TreeNode<K, V> node = pathToMin[--minDepth];
|
||||||
TreeNode node = (TreeNode)pathToMin[--minDepth];
|
|
||||||
node.left = right;
|
node.left = right;
|
||||||
right = node;
|
right = node;
|
||||||
node.balance();
|
node.balance();
|
||||||
|
@ -246,7 +381,10 @@ public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TS
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TSet<Entry<K, V>> entrySet() {
|
public TSet<Entry<K, V>> entrySet() {
|
||||||
return new EntrySet(new Object[0], null);
|
if (cachedEntrySet == null) {
|
||||||
|
cachedEntrySet = new EntrySet<>(this, null, true, false, null, true, false);
|
||||||
|
}
|
||||||
|
return cachedEntrySet;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -256,49 +394,58 @@ public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TS
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TSortedMap<K, V> subMap(K fromKey, K toKey) {
|
public TSortedMap<K, V> subMap(K fromKey, K toKey) {
|
||||||
return null;
|
if (comparator.compare(fromKey, toKey) > 0) {
|
||||||
|
throw new TIllegalArgumentException();
|
||||||
|
}
|
||||||
|
return new SubMap<>(this, fromKey, true, true, toKey, false, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TSortedMap<K, V> headMap(K toKey) {
|
public TSortedMap<K, V> headMap(K toKey) {
|
||||||
return null;
|
return new SubMap<>(this, null, true, false, toKey, false, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TSortedMap<K, V> tailMap(K fromKey) {
|
public TSortedMap<K, V> tailMap(K fromKey) {
|
||||||
return null;
|
return new SubMap<>(this, fromKey, false, true, null, false, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public K firstKey() {
|
public K firstKey() {
|
||||||
return firstEntry().key;
|
TreeNode<K, V> node = firstNode();
|
||||||
|
if (node == null) {
|
||||||
|
throw new TNoSuchElementException();
|
||||||
|
}
|
||||||
|
return node.getKey();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public K lastKey() {
|
public K lastKey() {
|
||||||
return lastEntry().key;
|
TreeNode<K, V> node = lastNode();
|
||||||
}
|
if (node == null) {
|
||||||
|
|
||||||
private TreeNode firstEntry() {
|
|
||||||
if (isEmpty()) {
|
|
||||||
throw new TNoSuchElementException();
|
throw new TNoSuchElementException();
|
||||||
}
|
}
|
||||||
TreeNode node = root;
|
return node.getKey();
|
||||||
while (node.left != null) {
|
|
||||||
node = node.left;
|
|
||||||
}
|
|
||||||
return node;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private TreeNode lastEntry() {
|
private TreeNode<K, V> firstNode() {
|
||||||
if (isEmpty()) {
|
TreeNode<K, V> node = root;
|
||||||
throw new TNoSuchElementException();
|
TreeNode<K, V> prev = null;
|
||||||
}
|
while (node != null) {
|
||||||
TreeNode node = root;
|
prev = node;
|
||||||
while (node.left != null) {
|
|
||||||
node = node.left;
|
node = node.left;
|
||||||
}
|
}
|
||||||
return node;
|
return prev;
|
||||||
|
}
|
||||||
|
|
||||||
|
private TreeNode<K, V> lastNode() {
|
||||||
|
TreeNode<K, V> node = root;
|
||||||
|
TreeNode<K, V> prev = null;
|
||||||
|
while (node != null) {
|
||||||
|
prev = node;
|
||||||
|
node = node.right;
|
||||||
|
}
|
||||||
|
return prev;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -306,28 +453,50 @@ public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TS
|
||||||
return root != null ? root.size : 0;
|
return root != null ? root.size : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
private class EntrySet extends TAbstractSet<Entry<K, V>> {
|
int height() {
|
||||||
private Object[] path;
|
return root != null ? root.height : 0;
|
||||||
private TreeNode from;
|
}
|
||||||
private TreeNode to;
|
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@Override
|
||||||
public EntrySet(Object[] path, TreeNode to) {
|
public Object clone() {
|
||||||
this.path = path;
|
TTreeMap<?, ?> copy = (TTreeMap<?, ?>)super.clone();
|
||||||
this.from = this.path.length > 0 ? (TreeNode)this.path[this.path.length - 1] : null;
|
copy.cachedEntrySet = null;
|
||||||
|
return copy;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class EntrySet<K, V> extends TAbstractSet<Entry<K, V>> {
|
||||||
|
private TTreeMap<K, V> owner;
|
||||||
|
private K from;
|
||||||
|
private boolean fromIncluded;
|
||||||
|
private boolean fromChecked;
|
||||||
|
private K to;
|
||||||
|
private boolean toIncluded;
|
||||||
|
private boolean toChecked;
|
||||||
|
|
||||||
|
public EntrySet(TTreeMap<K, V> owner, K from, boolean fromIncluded, boolean fromChecked,
|
||||||
|
K to, boolean toIncluded, boolean toChecked) {
|
||||||
|
this.owner = owner;
|
||||||
|
this.from = from;
|
||||||
|
this.fromIncluded = fromIncluded;
|
||||||
|
this.fromChecked = fromChecked;
|
||||||
this.to = to;
|
this.to = to;
|
||||||
|
this.toIncluded = toIncluded;
|
||||||
|
this.toChecked = toChecked;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int size() {
|
public int size() {
|
||||||
int size = TTreeMap.this.size();
|
int size = owner.size();
|
||||||
if (from != null && from.left != null) {
|
if (fromChecked) {
|
||||||
size -= from.left.size;
|
TreeNode<K, V> node = fromIncluded ? owner.findPrev(from) : owner.findExactOrPrev(from);
|
||||||
|
if (node != null) {
|
||||||
|
size -= node.size;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (to != null) {
|
if (toChecked) {
|
||||||
size -= 1;
|
TreeNode<K, V> node = toIncluded ? owner.findNext(to) : owner.findExactOrNext(to);
|
||||||
if (to.right != null) {
|
if (node != null) {
|
||||||
size -= to.right.size;
|
size -= node.size;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return size;
|
return size;
|
||||||
|
@ -335,50 +504,283 @@ public class TTreeMap<K, V> extends TAbstractMap<K, V> implements TCloneable, TS
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TIterator<Entry<K, V>> iterator() {
|
public TIterator<Entry<K, V>> iterator() {
|
||||||
return null;
|
TreeNode<K, V>[] fromPath;
|
||||||
|
if (fromChecked) {
|
||||||
|
fromPath = fromIncluded ? owner.pathToExactOrNext(from) : owner.pathToNext(from);
|
||||||
|
} else {
|
||||||
|
fromPath = owner.pathToFirst();
|
||||||
|
}
|
||||||
|
TreeNode<K, V> toEntry;
|
||||||
|
if (toChecked) {
|
||||||
|
toEntry = toIncluded ? owner.findExactOrPrev(to) : owner.findPrev(to);
|
||||||
|
} else {
|
||||||
|
toEntry = owner.lastNode();
|
||||||
|
}
|
||||||
|
return new EntryIterator<>(owner, fromPath, toEntry);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean contains(Object o) {
|
||||||
|
if (!(o instanceof Entry)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
Entry<?, ?> entry = (Entry<?, ?>)o;
|
||||||
|
Object key = entry.getKey();
|
||||||
|
if (from != null) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
int cmp = owner.comparator.compare((K)key, from);
|
||||||
|
if (fromIncluded ? cmp < 0 : cmp <= 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (to != null) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
int cmp = owner.comparator.compare((K)key, to);
|
||||||
|
if (toIncluded ? cmp > 0 : cmp >= 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
TreeNode<?, V> node = owner.findExact(key);
|
||||||
|
return node != null && node.equals(o);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEmpty() {
|
||||||
|
return from == to;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private class EntryIterator implements TIterator<Entry<K, V>> {
|
private static class EntryIterator<K, V> implements TIterator<Entry<K, V>> {
|
||||||
private int modCount = TTreeMap.this.modCount;
|
private int modCount;
|
||||||
private Object[] path = new Object[root != null ? root.height : 0];
|
private TTreeMap<K, V> owner;
|
||||||
private TreeNode last;
|
private TreeNode<K, V>[] path;
|
||||||
private TreeNode to;
|
private TreeNode<K, V> last;
|
||||||
private int pathLength;
|
private TreeNode<K, V> to;
|
||||||
public EntryIterator(Object[] path, TreeNode to) {
|
private int depth;
|
||||||
TreeNode node = root;
|
|
||||||
while (node != null) {
|
public EntryIterator(TTreeMap<K, V> owner, TreeNode<K, V>[] path, TreeNode<K, V> to) {
|
||||||
path[pathLength++] = node;
|
this.owner = owner;
|
||||||
node = node.left;
|
modCount = owner.modCount;
|
||||||
}
|
this.path = TArrays.copyOf(path, owner.root.height);
|
||||||
|
depth = path.length;
|
||||||
|
this.to = to;
|
||||||
}
|
}
|
||||||
@Override public boolean hasNext() {
|
|
||||||
return pathLength > 0;
|
@Override
|
||||||
|
public boolean hasNext() {
|
||||||
|
return depth > 0;
|
||||||
}
|
}
|
||||||
@SuppressWarnings("unchecked") @Override public Entry<K, V> next() {
|
|
||||||
if (modCount != TTreeMap.this.modCount) {
|
@Override
|
||||||
|
public Entry<K, V> next() {
|
||||||
|
if (modCount != owner.modCount) {
|
||||||
throw new TConcurrentModificationException();
|
throw new TConcurrentModificationException();
|
||||||
}
|
}
|
||||||
if (pathLength == 0) {
|
if (depth == 0) {
|
||||||
throw new TNoSuchElementException();
|
throw new TNoSuchElementException();
|
||||||
}
|
}
|
||||||
TreeNode node = (TreeNode)path[--pathLength];
|
TreeNode<K, V> node = path[--depth];
|
||||||
last = node;
|
last = node;
|
||||||
if (node.right != null) {
|
if (node.right != null) {
|
||||||
node = node.right;
|
node = node.right;
|
||||||
while (node != null) {
|
while (node != null) {
|
||||||
path[pathLength++] = node;
|
path[depth++] = node;
|
||||||
node = node.left;
|
node = node.left;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (last == to) {
|
||||||
|
depth = 0;
|
||||||
|
}
|
||||||
return last;
|
return last;
|
||||||
}
|
}
|
||||||
@Override public void remove() {
|
|
||||||
|
@Override
|
||||||
|
public void remove() {
|
||||||
|
if (modCount != owner.modCount) {
|
||||||
|
throw new TConcurrentModificationException();
|
||||||
|
}
|
||||||
if (last == null) {
|
if (last == null) {
|
||||||
throw new TNoSuchElementException();
|
throw new TNoSuchElementException();
|
||||||
}
|
}
|
||||||
deleteNode(root, last);
|
owner.root = owner.deleteNode(owner.root, last.getKey());
|
||||||
|
modCount = ++owner.modCount;
|
||||||
last = null;
|
last = null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static class SubMap<K, V> extends TAbstractMap<K, V> implements TSortedMap<K, V> {
|
||||||
|
private TTreeMap<K, V> owner;
|
||||||
|
private K from;
|
||||||
|
private boolean fromIncluded;
|
||||||
|
private boolean fromChecked;
|
||||||
|
private K to;
|
||||||
|
private boolean toIncluded;
|
||||||
|
private boolean toChecked;
|
||||||
|
private EntrySet<K, V> entrySetCache;
|
||||||
|
|
||||||
|
public SubMap(TTreeMap<K, V> owner, K from, boolean fromIncluded, boolean fromChecked,
|
||||||
|
K to, boolean toIncluded, boolean toChecked) {
|
||||||
|
this.owner = owner;
|
||||||
|
this.from = from;
|
||||||
|
this.fromIncluded = fromIncluded;
|
||||||
|
this.fromChecked = fromChecked;
|
||||||
|
this.to = to;
|
||||||
|
this.toIncluded = toIncluded;
|
||||||
|
this.toChecked = toChecked;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TSet<Entry<K, V>> entrySet() {
|
||||||
|
if (entrySetCache == null) {
|
||||||
|
entrySetCache = new EntrySet<>(owner, from, fromIncluded, fromChecked, to, toIncluded, toChecked);
|
||||||
|
}
|
||||||
|
return entrySetCache;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TComparator<? super K> comparator() {
|
||||||
|
return owner.originalComparator;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void checkKey(K key) {
|
||||||
|
if (fromChecked) {
|
||||||
|
int cmp = owner.comparator.compare(key, from);
|
||||||
|
if (fromIncluded ? cmp < 0 : cmp <= 0) {
|
||||||
|
throw new TIllegalArgumentException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (toChecked) {
|
||||||
|
int cmp = owner.comparator.compare(key, to);
|
||||||
|
if (fromIncluded ? cmp > 0 : cmp >= 0) {
|
||||||
|
throw new TIllegalArgumentException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
@Override
|
||||||
|
public V get(Object key) {
|
||||||
|
checkKey((K)key);
|
||||||
|
return owner.get(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
@Override
|
||||||
|
public V remove(Object key) {
|
||||||
|
checkKey((K)key);
|
||||||
|
return owner.remove(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public V put(K key, V value) {
|
||||||
|
checkKey(key);
|
||||||
|
return owner.put(key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int size() {
|
||||||
|
int size = owner.size();
|
||||||
|
if (fromChecked) {
|
||||||
|
TreeNode<K, V> node = fromIncluded ? owner.findPrev(from) : owner.findExactOrPrev(from);
|
||||||
|
if (node != null) {
|
||||||
|
size -= node.size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (toChecked) {
|
||||||
|
TreeNode<K, V> node = toIncluded ? owner.findNext(to) : owner.findExactOrNext(to);
|
||||||
|
if (node != null) {
|
||||||
|
size -= node.size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean containsKey(Object key) {
|
||||||
|
if (fromChecked) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
int cmp = owner.comparator.compare((K)key, from);
|
||||||
|
if (fromIncluded ? cmp < 0 : cmp <= 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (toChecked) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
int cmp = owner.comparator.compare((K)key, to);
|
||||||
|
if (fromIncluded ? cmp > 0 : cmp >= 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return owner.containsKey(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TSortedMap<K, V> subMap(K fromKey, K toKey) {
|
||||||
|
checkKey(fromKey);
|
||||||
|
checkKey(toKey);
|
||||||
|
return new SubMap<>(owner, fromKey, true, true, toKey, false, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TSortedMap<K, V> headMap(K toKey) {
|
||||||
|
checkKey(toKey);
|
||||||
|
return new SubMap<>(owner, from, fromIncluded, fromChecked, toKey, false, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TSortedMap<K, V> tailMap(K fromKey) {
|
||||||
|
checkKey(fromKey);
|
||||||
|
return new SubMap<>(owner, fromKey, true, true, to, toIncluded, toChecked);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public K firstKey() {
|
||||||
|
TreeNode<K, V> node = firstNode();
|
||||||
|
if (node == null) {
|
||||||
|
throw new TNoSuchElementException();
|
||||||
|
}
|
||||||
|
return node.getKey();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public K lastKey() {
|
||||||
|
TreeNode<K, V> node = lastNode();
|
||||||
|
if (node == null) {
|
||||||
|
throw new TNoSuchElementException();
|
||||||
|
}
|
||||||
|
return node.getKey();
|
||||||
|
}
|
||||||
|
|
||||||
|
private TreeNode<K, V> firstNode() {
|
||||||
|
TreeNode<K, V> node;
|
||||||
|
if (fromChecked) {
|
||||||
|
node = fromIncluded ? owner.findExactOrNext(from) : owner.findNext(from);
|
||||||
|
} else {
|
||||||
|
node = owner.firstNode();
|
||||||
|
}
|
||||||
|
if (toChecked) {
|
||||||
|
int cmp = owner.comparator.compare(node.getKey(), to);
|
||||||
|
if (toIncluded ? cmp > 0 : cmp >= 0) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
|
||||||
|
private TreeNode<K, V> lastNode() {
|
||||||
|
TreeNode<K, V> node;
|
||||||
|
if (toChecked) {
|
||||||
|
node = toIncluded ? owner.findExactOrPrev(to) : owner.findPrev(to);
|
||||||
|
} else {
|
||||||
|
node = owner.lastNode();
|
||||||
|
}
|
||||||
|
if (fromChecked) {
|
||||||
|
int cmp = owner.comparator.compare(node.getKey(), from);
|
||||||
|
if (fromIncluded ? cmp < 0 : cmp <= 0) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -441,22 +441,16 @@ public class TreeMapTest {
|
||||||
Map<String, String> subMap = map.subMap("", "test");
|
Map<String, String> subMap = map.subMap("", "test");
|
||||||
assertEquals(3, subMap.size());
|
assertEquals(3, subMap.size());
|
||||||
|
|
||||||
Set<Map.Entry<String, String>> entrySet = subMap.entrySet();
|
|
||||||
Iterator<Map.Entry<String, String>> iter = entrySet.iterator();
|
|
||||||
int size = 0;
|
int size = 0;
|
||||||
while (iter.hasNext()) {
|
for (Map.Entry<String, String> entry : subMap.entrySet()) {
|
||||||
Map.Entry<String, String> entry = iter.next();
|
|
||||||
assertTrue(map.containsKey(entry.getKey()));
|
assertTrue(map.containsKey(entry.getKey()));
|
||||||
assertTrue(map.containsValue(entry.getValue()));
|
assertTrue(map.containsValue(entry.getValue()));
|
||||||
size++;
|
size++;
|
||||||
}
|
}
|
||||||
assertEquals(map.size(), size);
|
assertEquals(map.size(), size);
|
||||||
|
|
||||||
Set<String> keySet = subMap.keySet();
|
|
||||||
Iterator<String> keyIter = keySet.iterator();
|
|
||||||
size = 0;
|
size = 0;
|
||||||
while (keyIter.hasNext()) {
|
for (String key : subMap.keySet()) {
|
||||||
String key = keyIter.next();
|
|
||||||
assertTrue(map.containsKey(key));
|
assertTrue(map.containsKey(key));
|
||||||
size++;
|
size++;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user