All Downloads are FREE. Search and download functionalities are using the official Maven repository.

oracle.toplink.essentials.internal.helper.TopLinkIdentityHashSet Maven / Gradle / Ivy

/*
 * The contents of this file are subject to the terms 
 * of the Common Development and Distribution License 
 * (the "License").  You may not use this file except 
 * in compliance with the License.
 * 
 * You can obtain a copy of the license at 
 * glassfish/bootstrap/legal/CDDLv1.0.txt or 
 * https://glassfish.dev.java.net/public/CDDLv1.0.html. 
 * See the License for the specific language governing 
 * permissions and limitations under the License.
 * 
 * When distributing Covered Code, include this CDDL 
 * HEADER in each file and include the License file at 
 * glassfish/bootstrap/legal/CDDLv1.0.txt.  If applicable, 
 * add the following below this CDDL HEADER, with the 
 * fields enclosed by brackets "[]" replaced with your 
 * own identifying information: Portions Copyright [yyyy] 
 * [name of copyright owner]
 */
// Copyright (c) 1998, 2007, Oracle. All rights reserved.  
package oracle.toplink.essentials.internal.helper;


/**
 * INTERNAL:
 * 

* Purpose: Define a {@link Set} that tests equality by reference, * not equals(). This is required to track objects throughout the lifecycle * of a {@link oracle.toplink.essentials.sessions.UnitOfWork}, regardless if the domain * object redefines its equals() method. Additionally, this implementation does * not allow null elements. *

* This class does not inherit from {@link AbstractSet} because the * method {@link AbstractSet#removeAll removeAll(Collection c)} does not work * correctly with reference equality testing (NB the Javadocs for * {@link AbstractCollection} indicates that removeAll is an optional method). * * @author Mike Norman (since TopLink 10.1.3) * */ // J2SE imports import java.io.*; import java.util.*; public class TopLinkIdentityHashSet extends AbstractCollection implements Set, Cloneable, Serializable { static final long serialVersionUID = 1619330892277906704L; // the default initial capacity static final int DEFAULT_INITIAL_CAPACITY = 32; // the maximum capacity. static final int MAXIMUM_CAPACITY = 1 << 30; // the loadFactor used when none specified in constructor. static final float DEFAULT_LOAD_FACTOR = 0.75f; protected transient Entry[] entries;// internal array of Entry's protected transient int count = 0; protected int threshold = 0; protected float loadFactor = 0; /** * Constructs a new TopLinkIdentityHashSet with the given initial * capacity and the given loadFactor. * * @param initialCapacity the initial capacity of the * TopLinkIdentityHashSet. * @param loadFactor the loadFactor of the TopLinkIdentityHashSet. * @throws IllegalArgumentException if the initial capacity is less * than zero, or if the loadFactor is nonpositive. */ public TopLinkIdentityHashSet(int initialCapacity, float loadFactor) { if (initialCapacity < 0) { throw new IllegalArgumentException("Illegal initialCapacity: " + initialCapacity); } if (initialCapacity > MAXIMUM_CAPACITY) { initialCapacity = MAXIMUM_CAPACITY; } if ((loadFactor <= 0) || Float.isNaN(loadFactor)) { throw new IllegalArgumentException("Illegal loadFactor: " + loadFactor); } // Find a power of 2 >= initialCapacity int capacity = 1; while (capacity < initialCapacity) { capacity <<= 1; } this.loadFactor = loadFactor; threshold = (int)(capacity * loadFactor); entries = new Entry[capacity]; } /** * Constructs a new TopLinkIdentityHashSet with the given * initial capacity and a default loadFactor of 0.75. * * @param initialCapacity the initial capacity of the TopLinkIdentityHashSet. * @throws IllegalArgumentException if the initial capacity is less * than zero. */ public TopLinkIdentityHashSet(int initialCapacity) { this(initialCapacity, DEFAULT_LOAD_FACTOR); } /** * Constructs a new TopLinkIdentityHashSet with a default initial * capacity of 32 and a loadfactor of 0.75. */ public TopLinkIdentityHashSet() { loadFactor = DEFAULT_LOAD_FACTOR; threshold = (int)(DEFAULT_INITIAL_CAPACITY * DEFAULT_LOAD_FACTOR); entries = new Entry[DEFAULT_INITIAL_CAPACITY]; } /** * Constructs a new TopLinkIdentityHashSet with the same contents * as the given Collection. The new TopLinkIdentityHashSet * is created with an initial capacity sufficient to hold the elements of * the given Collection. * * @param c the Collection whose contents are to be placed in the * new TopLinkIdentityHashSet. */ public TopLinkIdentityHashSet(Collection c) { this(Math.max((int)(c.size() / DEFAULT_LOAD_FACTOR) + 1, DEFAULT_INITIAL_CAPACITY), DEFAULT_LOAD_FACTOR); addAll(c); } /** * @return the size of this TopLinkIdentityHashSet. */ public int size() { return count; } /** * @return true if this TopLinkIdentityHashSet is empty. */ public boolean isEmpty() { return (count == 0); } /** * Returns true if this TopLinkIdentityHashSet contains * the given object. * * @param obj the object to find. * @return true if this TopLinkIdentityHashSet contains * obj by reference. */ public boolean contains(Object obj) { if (obj == null) { return false; } Entry[] copyOfEntries = entries; int hash = System.identityHashCode(obj); int index = (hash & 0x7FFFFFFF) % copyOfEntries.length; for (Entry e = copyOfEntries[index]; e != null; e = e.next) { if ((e.hash == hash) && (obj == e.value)) { return true; } } return false; } /** * INTERNAL: * Re-builds the internal array of Entry's with a larger capacity. * This method is called automatically when the number of objects in this * TopLinkIdentityHashSet exceeds its current threshold. */ private void rehash() { int oldCapacity = entries.length; Entry[] oldEntries = entries; int newCapacity = (oldCapacity * 2) + 1; Entry[] newEntries = new Entry[newCapacity]; threshold = (int)(newCapacity * loadFactor); entries = newEntries; for (int i = oldCapacity; i-- > 0;) { for (Entry old = oldEntries[i]; old != null;) { Entry e = old; old = old.next; int index = (e.hash & 0x7FFFFFFF) % newCapacity; e.next = newEntries[index]; newEntries[index] = e; } } } /** * Adds the given object to this TopLinkIdentityHashSet. * * @param obj object to add. * @return true if this TopLinkIdentityHashSet did not * already contain obj. * @throws NullPointerException if obj is null. */ public boolean add(Object obj) { if (obj == null) { throw new NullPointerException(); } // Makes sure the object is not already in the TopLinkIdentityHashSet. Entry[] copyOfEntries = entries; int hash = System.identityHashCode(obj); int index = (hash & 0x7FFFFFFF) % copyOfEntries.length; for (Entry e = copyOfEntries[index]; e != null; e = e.next) { if ((e.hash == hash) && (obj == e.value)) { return false; } } if (count >= threshold) { // Rehash the table if the threshold is exceeded rehash(); copyOfEntries = entries; index = (hash & 0x7FFFFFFF) % copyOfEntries.length; } // Creates the new entry. Entry e = new Entry(hash, obj, copyOfEntries[index]); copyOfEntries[index] = e; count++; return true; } /** * Removes the given object from this TopLinkIdentityHashSet, if * present. * * @param obj the object to be removed from this TopLinkIdentityHashSet. * @return true if this TopLinkIdentityHashSet contained * obj. */ public boolean remove(Object obj) { if (obj == null) { return false; } Entry[] copyOfEntries = entries; int hash = System.identityHashCode(obj); int index = (hash & 0x7FFFFFFF) % copyOfEntries.length; for (Entry e = copyOfEntries[index], prev = null; e != null; prev = e, e = e.next) { if ((e.hash == hash) && (obj == e.value)) { if (prev != null) { prev.next = e.next; } else { copyOfEntries[index] = e.next; } count--; return true; } } return false; } /** * This implementation throws an UnsupportedOperationException * because removeAll does not work correctly with reference * equality testing.

*/ public boolean removeAll(Collection c) { throw new UnsupportedOperationException("TopLinkIdentityHashSet removeAll"); } /** * This implementation throws an UnsupportedOperationException. * The Javadocs for {@link AbstractCollection} indicates that retainAll * is an optional method.

*/ public boolean retainAll(Collection c) { throw new UnsupportedOperationException("TopLinkIdentityHashSet retainAll"); } /** * Removes all of the objects from this TopLinkIdentityHashSet. */ public void clear() { if (count > 0) { Entry[] copyOfEntries = entries; for (int i = copyOfEntries.length; --i >= 0;) { copyOfEntries[i] = null; } count = 0; } } /** * Returns a shallow copy of this TopLinkIdentityHashSet (the * elements are not cloned). * * @return a shallow copy of this TopLinkIdentityHashSet. */ public Object clone() { try { Entry[] copyOfEntries = entries; TopLinkIdentityHashSet clone = (TopLinkIdentityHashSet)super.clone(); clone.entries = new Entry[copyOfEntries.length]; for (int i = copyOfEntries.length; i-- > 0;) { clone.entries[i] = (copyOfEntries[i] != null) ? (Entry)copyOfEntries[i].clone() : null; } return clone; } catch (CloneNotSupportedException e) { // this shouldn't happen, since we are Cloneable throw new InternalError(); } } /** * Get an iterator for this TopLinkIdentityHashSet */ public Iterator iterator() { return new TopLinkIdentityHashSetIterator(); } /** * TopLinkIdentityHashSet entry. */ static class Entry { int hash; Object value; Entry next; Entry(int hash, Object value, Entry next) { this.hash = hash; this.value = value; this.next = next; } protected Object clone() { return new Entry(hash, value, ((next == null) ? null : (Entry)next.clone())); } } class TopLinkIdentityHashSetIterator implements Iterator { Entry[] copyOfEntries = TopLinkIdentityHashSet.this.entries; int index = copyOfEntries.length; Entry entry = null; Entry lastReturned = null; TopLinkIdentityHashSetIterator() { } public boolean hasNext() { Entry e = entry; int i = index; Entry[] tmp = copyOfEntries; while ((e == null) && (i > 0)) { e = tmp[--i]; } entry = e; index = i; return e != null; } public Object next() { Entry et = entry; int i = index; Entry[] tmp = copyOfEntries; while ((et == null) && (i > 0)) { et = tmp[--i]; } entry = et; index = i; if (et != null) { Entry e = lastReturned = entry; entry = e.next; return e.value; } throw new NoSuchElementException(); } public void remove() { if (lastReturned == null) { throw new IllegalStateException(); } Entry[] copyOfEntries = TopLinkIdentityHashSet.this.entries; int index = (lastReturned.hash & 0x7FFFFFFF) % copyOfEntries.length; for (Entry e = copyOfEntries[index], prev = null; e != null; prev = e, e = e.next) { if (e == lastReturned) { if (prev == null) { copyOfEntries[index] = e.next; } else { prev.next = e.next; } count--; lastReturned = null; return; } } throw new ConcurrentModificationException(); } } /** * Serialize the state of this TopLinkIdentityHashSet to a stream. * * @serialData The capacity of the TopLinkIdentityHashSet * (the length of the bucket array) is emitted (int), followed by the * size of the TopLinkIdentityHashSet, followed by the * contents (in no particular order). */ private void writeObject(ObjectOutputStream s) throws IOException, ClassNotFoundException { // Write out the threshold, loadfactor (and any hidden 'magic' stuff). s.defaultWriteObject(); // Write out number of buckets s.writeInt(entries.length); // Write out count s.writeInt(count); // Write out contents for (Iterator i = iterator(); i.hasNext();) { s.writeObject(i.next()); } } /** * Deserialize the TopLinkIdentityHashSet from a stream. */ private void readObject(ObjectInputStream s) throws IOException, ClassNotFoundException { // Read in the threshold, loadfactor (and any hidden 'magic' stuff). s.defaultReadObject(); // Read in number of buckets and allocate the bucket array; int numBuckets = s.readInt(); entries = new Entry[numBuckets]; // Read in size (count) int size = s.readInt(); // Read the objects and add to the TopLinkIdentityHashSet for (int i = 0; i < size; i++) { add(s.readObject()); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy