1 /* Copyright (C) 2003 Vladimir Roubtsov. All rights reserved.
2  *
3  * This program and the accompanying materials are made available under
4  * the terms of the Common Public License v1.0 which accompanies this distribution,
5  * and is available at http://www.eclipse.org/legal/cpl-v10.html
6  *
7  * $Id: ObjectIntMap.java,v 1.1.1.1 2004/05/09 16:57:54 vlad_r Exp $
8  */
9 package com.vladium.util;
10 
11 import com.vladium.util.asserts.$assert;
12 
13 // ----------------------------------------------------------------------------
14 /**
15  *
16  * MT-safety: an instance of this class is <I>not</I> safe for access from
17  * multiple concurrent threads [even if access is done by a single thread at a
18  * time]. The caller is expected to synchronize externally on an instance [the
19  * implementation does not do internal synchronization for the sake of efficiency].
20  * java.util.ConcurrentModificationException is not supported either.
21  *
22  * @author (C) 2001, Vlad Roubtsov
23  */
24 public
25 final class ObjectIntMap
26 {
27     // public: ................................................................
28 
29     // TODO: optimize key comparisons using key.hash == entry.key.hash condition
30 
31     /**
32      * Equivalent to <CODE>IntObjectMap(11, 0.75F)</CODE>.
33      */
ObjectIntMap()34     public ObjectIntMap ()
35     {
36         this (11, 0.75F);
37     }
38 
39     /**
40      * Equivalent to <CODE>IntObjectMap(capacity, 0.75F)</CODE>.
41      */
ObjectIntMap(final int initialCapacity)42     public ObjectIntMap (final int initialCapacity)
43     {
44         this (initialCapacity, 0.75F);
45     }
46 
47     /**
48      * Constructs an IntObjectMap with specified initial capacity and load factor.
49      *
50      * @param initialCapacity initial number of hash buckets in the table [may not be negative, 0 is equivalent to 1].
51      * @param loadFactor the load factor to use to determine rehashing points [must be in (0.0, 1.0] range].
52      */
ObjectIntMap(int initialCapacity, final float loadFactor)53     public ObjectIntMap (int initialCapacity, final float loadFactor)
54     {
55         if (initialCapacity < 0) throw new IllegalArgumentException ("negative input: initialCapacity [" + initialCapacity + "]");
56         if ((loadFactor <= 0.0) || (loadFactor >= 1.0 + 1.0E-6))
57             throw new IllegalArgumentException ("loadFactor not in (0.0, 1.0] range: " + loadFactor);
58 
59         if (initialCapacity == 0) initialCapacity = 1;
60 
61         m_loadFactor = loadFactor > 1.0 ? 1.0F : loadFactor;
62         m_sizeThreshold = (int) (initialCapacity * loadFactor);
63         m_buckets = new Entry [initialCapacity];
64     }
65 
66 
67     /**
68      * Overrides Object.toString() for debug purposes.
69      */
toString()70     public String toString ()
71     {
72         final StringBuffer s = new StringBuffer ();
73         debugDump (s);
74 
75         return s.toString ();
76     }
77 
78     /**
79      * Returns the number of key-value mappings in this map.
80      */
size()81     public int size ()
82     {
83         return m_size;
84     }
85 
contains(final Object key)86     public boolean contains (final Object key)
87     {
88         if ($assert.ENABLED) $assert.ASSERT (key != null, "null input: key");
89 
90         // index into the corresponding hash bucket:
91         final Entry [] buckets = m_buckets;
92         final int keyHash = key.hashCode ();
93         final int bucketIndex = (keyHash & 0x7FFFFFFF) % buckets.length;
94 
95         // traverse the singly-linked list of entries in the bucket:
96         for (Entry entry = buckets [bucketIndex]; entry != null; entry = entry.m_next)
97         {
98             if ((keyHash == entry.m_key.hashCode ()) || entry.m_key.equals (key))
99                 return true;
100         }
101 
102         return false;
103     }
104 
105     /**
106      * Returns the value that is mapped to a given 'key'. Returns
107      * false if this key has never been mapped.
108      *
109      * @param key mapping key [may not be null]
110      * @param out holder for the found value [must be at least of size 1]
111      *
112      * @return 'true' if this key was mapped to an existing value
113      */
get(final Object key, final int [] out)114     public boolean get (final Object key, final int [] out)
115     {
116         if ($assert.ENABLED) $assert.ASSERT (key != null, "null input: key");
117 
118         // index into the corresponding hash bucket:
119         final Entry [] buckets = m_buckets;
120         final int keyHash = key.hashCode ();
121         final int bucketIndex = (keyHash & 0x7FFFFFFF) % buckets.length;
122 
123         // traverse the singly-linked list of entries in the bucket:
124         for (Entry entry = buckets [bucketIndex]; entry != null; entry = entry.m_next)
125         {
126             if ((keyHash == entry.m_key.hashCode ()) || entry.m_key.equals (key))
127             {
128                 out [0] = entry.m_value;
129                 return true;
130             }
131         }
132 
133         return false;
134     }
135 
keys()136     public Object [] keys ()
137     {
138         final Object [] result = new Object [m_size];
139         int scan = 0;
140 
141         for (int b = 0; b < m_buckets.length; ++ b)
142         {
143             for (Entry entry = m_buckets [b]; entry != null; entry = entry.m_next)
144             {
145                 result [scan ++] = entry.m_key;
146             }
147         }
148 
149         return result;
150     }
151 
152     /**
153      * Updates the table to map 'key' to 'value'. Any existing mapping is overwritten.
154      *
155      * @param key mapping key [may not be null]
156      * @param value mapping value.
157      */
put(final Object key, final int value)158     public void put (final Object key, final int value)
159     {
160         if ($assert.ENABLED) $assert.ASSERT (key != null, "null input: key");
161 
162         Entry currentKeyEntry = null;
163 
164         // detect if 'key' is already in the table [in which case, set 'currentKeyEntry' to point to its entry]:
165 
166         // index into the corresponding hash bucket:
167         final int keyHash = key.hashCode ();
168         int bucketIndex = (keyHash & 0x7FFFFFFF) % m_buckets.length;
169 
170         // traverse the singly-linked list of entries in the bucket:
171         Entry [] buckets = m_buckets;
172         for (Entry entry = buckets [bucketIndex]; entry != null; entry = entry.m_next)
173         {
174             if ((keyHash == entry.m_key.hashCode ()) || entry.m_key.equals (key))
175             {
176                 currentKeyEntry = entry;
177                 break;
178             }
179         }
180 
181         if (currentKeyEntry != null)
182         {
183             // replace the current value:
184 
185             currentKeyEntry.m_value = value;
186         }
187         else
188         {
189             // add a new entry:
190 
191             if (m_size >= m_sizeThreshold) rehash ();
192 
193             buckets = m_buckets;
194             bucketIndex = (keyHash & 0x7FFFFFFF) % buckets.length;
195             final Entry bucketListHead = buckets [bucketIndex];
196             final Entry newEntry = new Entry (key, value, bucketListHead);
197             buckets [bucketIndex] = newEntry;
198 
199             ++ m_size;
200         }
201     }
202 
203     /**
204      * Updates the table to map 'key' to 'value'. Any existing mapping is overwritten.
205      *
206      * @param key mapping key [may not be null]
207      */
remove(final Object key)208     public void remove (final Object key)
209     {
210         if ($assert.ENABLED) $assert.ASSERT (key != null, "null input: key");
211 
212         // index into the corresponding hash bucket:
213         final int keyHash = key.hashCode ();
214         final int bucketIndex = (keyHash  & 0x7FFFFFFF) % m_buckets.length;
215 
216         // traverse the singly-linked list of entries in the bucket:
217         Entry [] buckets = m_buckets;
218         for (Entry entry = buckets [bucketIndex], prev = entry; entry != null; )
219         {
220             final Entry next = entry.m_next;
221 
222             if ((keyHash == entry.m_key.hashCode ()) || entry.m_key.equals (key))
223             {
224                 if (prev == entry)
225                     buckets [bucketIndex] = next;
226                 else
227                     prev.m_next = next;
228 
229                 -- m_size;
230                 break;
231             }
232 
233             prev = entry;
234             entry = next;
235         }
236     }
237 
238 
239     // protected: .............................................................
240 
241     // package: ...............................................................
242 
243 
debugDump(final StringBuffer out)244     void debugDump (final StringBuffer out)
245     {
246         if (out != null)
247         {
248             out.append (super.toString ()); out.append (EOL);
249             out.append ("size = " + m_size + ", bucket table size = " + m_buckets.length + ", load factor = " + m_loadFactor + EOL);
250             out.append ("size threshold = " + m_sizeThreshold + EOL);
251         }
252     }
253 
254     // private: ...............................................................
255 
256 
257     /**
258      * The structure used for chaining colliding keys.
259      */
260     private static final class Entry
261     {
Entry(final Object key, final int value, final Entry next)262         Entry (final Object key, final int value, final Entry next)
263         {
264             m_key = key;
265             m_value = value;
266             m_next = next;
267         }
268 
269         Object m_key;     // reference to the value [never null]
270         int m_value;
271 
272         Entry m_next; // singly-linked list link
273 
274     } // end of nested class
275 
276 
277     /**
278      * Re-hashes the table into a new array of buckets.
279      */
rehash()280     private void rehash ()
281     {
282         // TODO: it is possible to run this method twice, first time using the 2*k+1 prime sequencer for newBucketCount
283         // and then with that value reduced to actually shrink capacity. As it is right now, the bucket table can
284         // only grow in size
285 
286         final Entry [] buckets = m_buckets;
287 
288         final int newBucketCount = (m_buckets.length << 1) + 1;
289         final Entry [] newBuckets = new Entry [newBucketCount];
290 
291         // rehash all entry chains in every bucket:
292         for (int b = 0; b < buckets.length; ++ b)
293         {
294             for (Entry entry = buckets [b]; entry != null; )
295             {
296                 final Entry next = entry.m_next; // remember next pointer because we are going to reuse this entry
297                 final int entryKeyHash = entry.m_key.hashCode () & 0x7FFFFFFF;
298 
299                 // index into the corresponding new hash bucket:
300                 final int newBucketIndex = entryKeyHash % newBucketCount;
301 
302                 final Entry bucketListHead = newBuckets [newBucketIndex];
303                 entry.m_next = bucketListHead;
304                 newBuckets [newBucketIndex] = entry;
305 
306                 entry = next;
307             }
308         }
309 
310 
311         m_sizeThreshold = (int) (newBucketCount * m_loadFactor);
312         m_buckets = newBuckets;
313     }
314 
315 
316     private final float m_loadFactor; // determines the setting of m_sizeThreshold
317 
318     private Entry [] m_buckets; // table of buckets
319     private int m_size; // number of keys in the table, not cleared as of last check
320     private int m_sizeThreshold; // size threshold for rehashing
321 
322     private static final String EOL = System.getProperty ("line.separator", "\n");
323 
324 } // end of class
325 // ----------------------------------------------------------------------------
326 
327