|         /* UnifyHash Copyright (C) 1999-2002 Jochen Hoenicke.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation; either version 2, or (at your option)
 * any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this program; see the file COPYING.LESSER.  If not, write to
 * the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
 *
 * $Id: UnifyHash.java.in,v 1.3.2.2 2002/05/28 17:34:24 hoenicke Exp $
 */
 
 ///#ifdef JDK12
 import java.lang.ref.WeakReference;
 import java.lang.ref.ReferenceQueue;
 ///#endif
 
 import java.util.Comparator;
 import java.util.AbstractCollection;
 import java.util.Iterator;
 import java.util.NoSuchElementException;
 import java.util.ConcurrentModificationException;
 import java.lang.UnsupportedOperationException;
 
 public class UnifyHash extends AbstractCollection {
 /**
 * the default capacity
 */
 private static final int DEFAULT_CAPACITY = 11;
 
 /** the default load factor of a HashMap */
 private static final float DEFAULT_LOAD_FACTOR = 0.75F;
 
 ///#ifdef JDK12
 private ReferenceQueue queue = new ReferenceQueue();
 ///#endif
 
 static class Bucket
 ///#ifdef JDK12
 extends WeakReference
 ///#endif
 {
 ///#ifdef JDK12
 public Bucket(Object o, ReferenceQueue q) {
 super(o, q);
 }
 ///#else
 /// public Bucket(Object o) {
 ///     this.obj = o;
 /// }
 ///
 /// Object obj;
 ///
 /// public Object get() {
 ///     return obj;
 /// }
 ///#endif
 
 int hash;
 Bucket next;
 }
 
 private Bucket[] buckets;
 int modCount = 0;
 int size = 0;
 int threshold;
 float loadFactor;
 
 public UnifyHash(int initialCapacity, float loadFactor) {
 this.loadFactor = loadFactor;
 buckets = new Bucket[initialCapacity];
 threshold = (int) (loadFactor * initialCapacity);
 }
 
 public UnifyHash(int initialCapacity) {
 this(initialCapacity, DEFAULT_LOAD_FACTOR);
 }
 
 public UnifyHash() {
 this(DEFAULT_CAPACITY, DEFAULT_LOAD_FACTOR);
 }
 
 private void grow() {
 Bucket[] oldBuckets = buckets;
 int newCap = buckets.length * 2 + 1;
 threshold = (int) (loadFactor * newCap);
 buckets = new Bucket[newCap];
 for (int i = 0; i < oldBuckets.length; i++) {
 Bucket nextBucket;
 for (Bucket b = oldBuckets[i]; b != null; b = nextBucket) {
 if (i != Math.abs(b.hash % oldBuckets.length))
 throw new RuntimeException(""+i+", hash: "+b.hash+", oldlength: "+oldBuckets.length);
 int newSlot = Math.abs(b.hash % newCap);
 nextBucket = b.next;
 b.next = buckets[newSlot];
 buckets[newSlot] = b;
 }
 }
 }
 
 ///#ifdef JDK12
 public final void cleanUp() {
 Bucket died;
 while ((died = (Bucket)queue.poll()) != null) {
 int diedSlot = Math.abs(died.hash % buckets.length);
 if (buckets[diedSlot] == died)
 buckets[diedSlot] = died.next;
 else {
 Bucket b = buckets[diedSlot];
 while (b.next != died)
 b = b.next;
 b.next = died.next;
 }
 size--;
 }
 }
 ///#endif
 
 
 public int size() {
 return size;
 }
 
 public Iterator iterator() {
 ///#ifdef JDK12
 cleanUp();
 ///#endif
 
 return new Iterator() {
 private int bucket = 0;
 private int known = modCount;
 private Bucket nextBucket;
 private Object nextVal;
 
 {
 internalNext();
 }
 
 private void internalNext() {
 while (true) {
 while (nextBucket == null) {
 if (bucket == buckets.length)
 return;
 nextBucket = buckets[bucket++];
 }
 
 nextVal = nextBucket.get();
 if (nextVal != null)
 return;
 
 nextBucket = nextBucket.next;
 }
 }
 
 public boolean hasNext() {
 return nextBucket != null;
 }
 
 public Object next() {
 if (known != modCount)
 throw new ConcurrentModificationException();
 if (nextBucket == null)
 throw new NoSuchElementException();
 Object result = nextVal;
 nextBucket = nextBucket.next;
 internalNext();
 return result;
 }
 
 public void remove() {
 throw new UnsupportedOperationException();
 }
 };
 }
 
 public Iterator iterateHashCode(final int hash) {
 ///#ifdef JDK12
 cleanUp();
 ///#endif
 return new Iterator() {
 private int known = modCount;
 private Bucket nextBucket
 = buckets[Math.abs(hash % buckets.length)];
 private Object nextVal;
 
 {
 internalNext();
 }
 
 private void internalNext() {
 while (nextBucket != null) {
 if (nextBucket.hash == hash) {
 nextVal = nextBucket.get();
 if (nextVal != null)
 return;
 }
 
 nextBucket = nextBucket.next;
 }
 }
 
 public boolean hasNext() {
 return nextBucket != null;
 }
 
 public Object next() {
 if (known != modCount)
 throw new ConcurrentModificationException();
 if (nextBucket == null)
 throw new NoSuchElementException();
 Object result = nextVal;
 nextBucket = nextBucket.next;
 internalNext();
 return result;
 }
 
 public void remove() {
 throw new UnsupportedOperationException();
 }
 };
 }
 
 public void put(int hash, Object o) {
 if (size++ > threshold)
 grow();
 modCount++;
 
 int slot = Math.abs(hash % buckets.length);
 ///#ifdef JDK12
 Bucket b = new Bucket(o, queue);
 ///#else
 /// Bucket b = new Bucket(o);
 ///#endif
 b.hash = hash;
 b.next = buckets[slot];
 buckets[slot] = b;
 }
 
 public Object unify(Object o, int hash, Comparator comparator) {
 ///#ifdef JDK12
 cleanUp();
 ///#endif
 int slot = Math.abs(hash % buckets.length);
 for (Bucket b = buckets[slot]; b != null; b = b.next) {
 Object old = b.get();
 if (old != null && comparator.compare(o, old) == 0)
 return old;
 }
 
 put(hash, o);
 return o;
 }
 }
 
 
 
 
 
 
 
 
 
 
 |