2007-07-21 20:44:39 +00:00
|
|
|
package java.util;
|
|
|
|
|
|
|
|
public class HashMap<K, V> implements Map<K, V> {
|
2007-08-21 00:24:54 +00:00
|
|
|
private static final int MinimumCapacity = 16;
|
|
|
|
|
2007-07-21 20:44:39 +00:00
|
|
|
private int size;
|
|
|
|
private Cell[] array;
|
2007-08-12 21:01:47 +00:00
|
|
|
private final Helper helper;
|
2007-07-21 20:44:39 +00:00
|
|
|
|
2007-08-15 01:14:55 +00:00
|
|
|
public HashMap(int capacity, Helper<K, V> helper) {
|
2007-07-21 20:44:39 +00:00
|
|
|
if (capacity > 0) {
|
|
|
|
array = new Cell[nextPowerOfTwo(capacity)];
|
|
|
|
}
|
2007-08-12 21:01:47 +00:00
|
|
|
this.helper = helper;
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public HashMap(int capacity) {
|
2007-08-12 21:01:47 +00:00
|
|
|
this(capacity, new MyHelper());
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public HashMap() {
|
|
|
|
this(0);
|
|
|
|
}
|
|
|
|
|
2007-08-15 01:14:55 +00:00
|
|
|
public String toString() {
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
sb.append("{");
|
|
|
|
for (Iterator<Entry<K, V>> it = iterator(); it.hasNext();) {
|
|
|
|
Entry<K, V> e = it.next();
|
|
|
|
sb.append(e.getKey())
|
|
|
|
.append("=")
|
|
|
|
.append(e.getValue());
|
|
|
|
if (it.hasNext()) {
|
|
|
|
sb.append(",");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sb.append("}");
|
|
|
|
return sb.toString();
|
|
|
|
}
|
|
|
|
|
2007-07-21 20:44:39 +00:00
|
|
|
private static int nextPowerOfTwo(int n) {
|
|
|
|
int r = 1;
|
|
|
|
while (r < n) r <<= 1;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int size() {
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2007-08-21 00:24:54 +00:00
|
|
|
private void grow() {
|
2007-07-21 20:44:39 +00:00
|
|
|
if (array == null || size >= array.length * 2) {
|
2007-08-21 00:24:54 +00:00
|
|
|
resize(array == null ? MinimumCapacity : array.length * 2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void shrink() {
|
|
|
|
if (array.length / 2 >= MinimumCapacity && size <= array.length / 3) {
|
2007-07-21 20:44:39 +00:00
|
|
|
resize(array.length / 2);
|
2007-08-21 00:24:54 +00:00
|
|
|
} else if (size == 0) {
|
|
|
|
resize(0);
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void resize(int capacity) {
|
|
|
|
Cell<K, V>[] newArray = null;
|
|
|
|
if (capacity != 0) {
|
|
|
|
capacity = nextPowerOfTwo(capacity);
|
|
|
|
if (array != null && array.length == capacity) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
newArray = new Cell[capacity];
|
|
|
|
if (array != null) {
|
|
|
|
for (int i = 0; i < array.length; ++i) {
|
|
|
|
Cell<K, V> next;
|
|
|
|
for (Cell<K, V> c = array[i]; c != null; c = next) {
|
|
|
|
next = c.next();
|
2007-08-14 00:37:00 +00:00
|
|
|
int index = c.hashCode() & (capacity - 1);
|
2007-08-14 01:44:47 +00:00
|
|
|
c.setNext(newArray[index]);
|
|
|
|
newArray[index] = c;
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
array = newArray;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Cell<K, V> find(K key) {
|
2007-07-22 19:06:21 +00:00
|
|
|
if (array != null) {
|
2007-08-13 00:50:25 +00:00
|
|
|
int index = helper.hash(key) & (array.length - 1);
|
2007-07-22 19:06:21 +00:00
|
|
|
for (Cell<K, V> c = array[index]; c != null; c = c.next()) {
|
2007-08-13 00:50:25 +00:00
|
|
|
if (helper.equal(key, c.getKey())) {
|
2007-07-22 19:06:21 +00:00
|
|
|
return c;
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-07-22 19:06:21 +00:00
|
|
|
|
|
|
|
return null;
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private void insert(Cell<K, V> cell) {
|
|
|
|
++ size;
|
|
|
|
|
2007-08-21 00:24:54 +00:00
|
|
|
grow();
|
2007-07-21 20:44:39 +00:00
|
|
|
|
2007-07-22 19:06:21 +00:00
|
|
|
int index = cell.hashCode() & (array.length - 1);
|
|
|
|
cell.setNext(array[index]);
|
|
|
|
array[index] = cell;
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
|
2007-08-15 01:14:55 +00:00
|
|
|
public void remove(Cell<K, V> cell) {
|
2007-07-22 19:06:21 +00:00
|
|
|
int index = cell.hashCode() & (array.length - 1);
|
|
|
|
Cell<K, V> p = null;
|
|
|
|
for (Cell<K, V> c = array[index]; c != null; c = c.next()) {
|
|
|
|
if (c == cell) {
|
|
|
|
if (p == null) {
|
|
|
|
array[index] = c.next();
|
|
|
|
} else {
|
|
|
|
p.setNext(c.next());
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
2007-07-22 19:06:21 +00:00
|
|
|
-- size;
|
|
|
|
break;
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
}
|
2007-07-22 19:06:21 +00:00
|
|
|
|
2007-08-21 00:24:54 +00:00
|
|
|
shrink();
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
|
2007-07-21 22:36:51 +00:00
|
|
|
private Cell<K, V> putCell(K key, V value) {
|
2007-07-21 20:44:39 +00:00
|
|
|
Cell<K, V> c = find(key);
|
|
|
|
if (c == null) {
|
2007-08-12 21:01:47 +00:00
|
|
|
insert(helper.make(key, value, null));
|
2007-07-21 20:44:39 +00:00
|
|
|
} else {
|
|
|
|
V old = c.getValue();
|
|
|
|
c.setValue(value);
|
|
|
|
}
|
2007-07-21 22:36:51 +00:00
|
|
|
return c;
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
|
2007-07-22 19:06:21 +00:00
|
|
|
public boolean containsKey(K key) {
|
|
|
|
return find(key) != null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean containsValue(V value) {
|
|
|
|
return values().contains(value);
|
|
|
|
}
|
|
|
|
|
2007-07-21 22:36:51 +00:00
|
|
|
public V get(K key) {
|
|
|
|
Cell<K, V> c = find(key);
|
|
|
|
return (c == null ? null : c.getValue());
|
|
|
|
}
|
|
|
|
|
|
|
|
public Cell<K, V> removeCell(K key) {
|
|
|
|
Cell<K, V> old = null;
|
2007-07-22 19:06:21 +00:00
|
|
|
if (array != null) {
|
2007-08-13 00:50:25 +00:00
|
|
|
int index = helper.hash(key) & (array.length - 1);
|
2007-07-22 19:06:21 +00:00
|
|
|
Cell<K, V> p = null;
|
|
|
|
for (Cell<K, V> c = array[index]; c != null; c = c.next()) {
|
2007-08-13 00:50:25 +00:00
|
|
|
if (helper.equal(key, c.getKey())) {
|
2007-07-22 19:06:21 +00:00
|
|
|
old = c;
|
|
|
|
if (p == null) {
|
|
|
|
array[index] = c.next();
|
|
|
|
} else {
|
|
|
|
p.setNext(c.next());
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
2007-07-22 19:06:21 +00:00
|
|
|
-- size;
|
|
|
|
break;
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
}
|
2007-07-22 19:06:21 +00:00
|
|
|
|
2007-08-21 00:24:54 +00:00
|
|
|
shrink();
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
return old;
|
|
|
|
}
|
|
|
|
|
2007-07-21 22:36:51 +00:00
|
|
|
public V put(K key, V value) {
|
|
|
|
Cell<K, V> c = putCell(key, value);
|
|
|
|
return (c == null ? null : c.getValue());
|
|
|
|
}
|
|
|
|
|
|
|
|
public V remove(K key) {
|
|
|
|
Cell<K, V> c = removeCell(key);
|
|
|
|
return (c == null ? null : c.getValue());
|
|
|
|
}
|
|
|
|
|
|
|
|
public void clear() {
|
|
|
|
array = null;
|
2007-07-22 03:47:08 +00:00
|
|
|
size = 0;
|
2007-07-21 22:36:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public Set<Entry<K, V>> entrySet() {
|
2007-07-22 03:47:08 +00:00
|
|
|
return new EntrySet();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Set<K> keySet() {
|
|
|
|
return new KeySet();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Collection<V> values() {
|
|
|
|
return new Values();
|
2007-07-21 22:36:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Iterator<Entry<K, V>> iterator() {
|
|
|
|
return new MyIterator();
|
|
|
|
}
|
|
|
|
|
2007-07-21 20:44:39 +00:00
|
|
|
interface Cell<K, V> extends Entry<K, V> {
|
|
|
|
public HashMap.Cell<K, V> next();
|
|
|
|
|
|
|
|
public void setNext(HashMap.Cell<K, V> next);
|
|
|
|
}
|
|
|
|
|
2007-08-12 21:01:47 +00:00
|
|
|
interface Helper<K, V> {
|
2007-07-21 20:44:39 +00:00
|
|
|
public Cell<K, V> make(K key, V value, Cell<K, V> next);
|
2007-08-12 21:01:47 +00:00
|
|
|
|
|
|
|
public int hash(K key);
|
|
|
|
|
|
|
|
public boolean equal(K a, K b);
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
|
2007-08-13 00:50:25 +00:00
|
|
|
private static class MyCell<K, V> implements Cell<K, V> {
|
2007-07-21 20:44:39 +00:00
|
|
|
public final K key;
|
|
|
|
public V value;
|
|
|
|
public Cell<K, V> next;
|
2007-08-13 00:50:25 +00:00
|
|
|
public int hashCode;
|
2007-07-21 20:44:39 +00:00
|
|
|
|
2007-08-13 00:50:25 +00:00
|
|
|
public MyCell(K key, V value, Cell<K, V> next, int hashCode) {
|
2007-07-21 20:44:39 +00:00
|
|
|
this.key = key;
|
|
|
|
this.value = value;
|
|
|
|
this.next = next;
|
2007-08-13 00:50:25 +00:00
|
|
|
this.hashCode = hashCode;
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public K getKey() {
|
|
|
|
return key;
|
|
|
|
}
|
|
|
|
|
|
|
|
public V getValue() {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setValue(V value) {
|
|
|
|
this.value = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
public HashMap.Cell<K, V> next() {
|
|
|
|
return next;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setNext(HashMap.Cell<K, V> next) {
|
|
|
|
this.next = next;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int hashCode() {
|
2007-08-13 00:50:25 +00:00
|
|
|
return hashCode;
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-08-12 21:01:47 +00:00
|
|
|
static class MyHelper<K, V> implements Helper<K, V> {
|
2007-07-21 20:44:39 +00:00
|
|
|
public Cell<K, V> make(K key, V value, Cell<K, V> next) {
|
2007-08-13 00:50:25 +00:00
|
|
|
return new MyCell(key, value, next, hash(key));
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
2007-08-12 21:01:47 +00:00
|
|
|
|
|
|
|
public int hash(K a) {
|
|
|
|
return (a == null ? 0 : a.hashCode());
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean equal(K a, K b) {
|
|
|
|
return (a == null && b == null) || (a != null && a.equals(b));
|
|
|
|
}
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|
2007-07-21 22:36:51 +00:00
|
|
|
|
2007-07-22 03:47:08 +00:00
|
|
|
private class EntrySet implements Set<Entry<K, V>> {
|
2007-07-21 22:36:51 +00:00
|
|
|
public int size() {
|
|
|
|
return HashMap.this.size();
|
|
|
|
}
|
|
|
|
|
2007-07-22 19:06:21 +00:00
|
|
|
public boolean contains(Entry<K, V> e) {
|
|
|
|
return containsKey(e.getKey());
|
|
|
|
}
|
|
|
|
|
2007-07-21 22:36:51 +00:00
|
|
|
public boolean add(Entry<K, V> e) {
|
|
|
|
return putCell(e.getKey(), e.getValue()) != null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean remove(Entry<K, V> e) {
|
|
|
|
return removeCell(e.getKey()) != null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void clear() {
|
|
|
|
HashMap.this.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Iterator<Entry<K, V>> iterator() {
|
|
|
|
return new MyIterator();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-22 03:47:08 +00:00
|
|
|
private class KeySet implements Set<K> {
|
|
|
|
public int size() {
|
|
|
|
return HashMap.this.size();
|
|
|
|
}
|
|
|
|
|
2007-07-22 19:06:21 +00:00
|
|
|
public boolean contains(K key) {
|
|
|
|
return containsKey(key);
|
|
|
|
}
|
|
|
|
|
2007-07-22 03:47:08 +00:00
|
|
|
public boolean add(K key) {
|
|
|
|
return putCell(key, null) != null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean remove(K key) {
|
|
|
|
return removeCell(key) != null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void clear() {
|
|
|
|
HashMap.this.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Iterator<K> iterator() {
|
|
|
|
return new KeyIterator(new MyIterator());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private class Values implements Collection<V> {
|
|
|
|
public int size() {
|
|
|
|
return HashMap.this.size();
|
|
|
|
}
|
|
|
|
|
2007-07-22 19:06:21 +00:00
|
|
|
public boolean contains(V value) {
|
|
|
|
return containsValue(value);
|
|
|
|
}
|
|
|
|
|
2007-07-22 03:47:08 +00:00
|
|
|
public boolean add(V value) {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean remove(V value) {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void clear() {
|
|
|
|
HashMap.this.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Iterator<V> iterator() {
|
|
|
|
return new ValueIterator(new MyIterator());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-21 22:36:51 +00:00
|
|
|
private class MyIterator implements Iterator<Entry<K, V>> {
|
|
|
|
private int currentIndex = -1;
|
|
|
|
private int nextIndex = -1;
|
|
|
|
private Cell<K, V> previousCell;
|
|
|
|
private Cell<K, V> currentCell;
|
|
|
|
private Cell<K, V> nextCell;
|
|
|
|
|
|
|
|
public MyIterator() {
|
|
|
|
hasNext();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Entry<K, V> next() {
|
|
|
|
if (hasNext()) {
|
|
|
|
if (currentCell != null && currentCell.next() != null) {
|
|
|
|
previousCell = currentCell;
|
|
|
|
} else {
|
|
|
|
previousCell = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
currentCell = nextCell;
|
|
|
|
currentIndex = nextIndex;
|
|
|
|
|
|
|
|
nextCell = nextCell.next();
|
|
|
|
|
|
|
|
return currentCell;
|
|
|
|
} else {
|
|
|
|
throw new NoSuchElementException();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean hasNext() {
|
|
|
|
if (array != null) {
|
|
|
|
while (nextCell == null && ++ nextIndex < array.length) {
|
|
|
|
if (array[nextIndex] != null) {
|
|
|
|
nextCell = array[nextIndex];
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-08-15 01:14:55 +00:00
|
|
|
return nextCell != null;
|
2007-07-21 22:36:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void remove() {
|
|
|
|
if (currentCell != null) {
|
|
|
|
if (previousCell == null) {
|
|
|
|
array[currentIndex] = currentCell.next();
|
|
|
|
} else {
|
|
|
|
previousCell.setNext(currentCell.next());
|
|
|
|
if (previousCell.next() == null) {
|
|
|
|
previousCell = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
currentCell = null;
|
|
|
|
} else {
|
|
|
|
throw new IllegalStateException();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-07-22 03:47:08 +00:00
|
|
|
|
|
|
|
private static class KeyIterator<K, V> implements Iterator<K> {
|
|
|
|
private final Iterator<Entry<K, V>> it;
|
|
|
|
|
|
|
|
public KeyIterator(Iterator<Entry<K, V>> it) {
|
|
|
|
this.it = it;
|
|
|
|
}
|
|
|
|
|
|
|
|
public K next() {
|
|
|
|
return it.next().getKey();
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean hasNext() {
|
|
|
|
return it.hasNext();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void remove() {
|
|
|
|
it.remove();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static class ValueIterator<K, V> implements Iterator<V> {
|
|
|
|
private final Iterator<Entry<K, V>> it;
|
|
|
|
|
|
|
|
public ValueIterator(Iterator<Entry<K, V>> it) {
|
|
|
|
this.it = it;
|
|
|
|
}
|
|
|
|
|
|
|
|
public V next() {
|
|
|
|
return it.next().getValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean hasNext() {
|
|
|
|
return it.hasNext();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void remove() {
|
|
|
|
it.remove();
|
|
|
|
}
|
|
|
|
}
|
2007-07-21 20:44:39 +00:00
|
|
|
}
|