view tango/tango/util/collection/HashMap.d @ 341:1bb99290e03a trunk

[svn r362] Started merging the old 'test' dir as well as the newer 'tangotests' dir into 'tests/mini' and 'tests/minicomplex'.
author lindquist
date Sun, 13 Jul 2008 02:51:19 +0200
parents 1700239cab2e
children
line wrap: on
line source

/*******************************************************************************

        File: HashMap.d

        Originally written by Doug Lea and released into the public domain. 
        Thanks for the assistance and support of Sun Microsystems Labs, Agorics 
        Inc, Loral, and everyone contributing, testing, and using this code.

        History:
        Date     Who                What
        24Sep95  dl@cs.oswego.edu   Create from collection.d  working file
        13Oct95  dl                 Changed protection statuses
        21Oct95  dl                 fixed error in removeAt
        9Apr97   dl                 made Serializable
        14Dec06  kb                 Converted, templated & reshaped for Tango
        
********************************************************************************/

module tango.util.collection.HashMap;

private import  tango.core.Exception;

/+
private import  tango.io.protocol.model.IReader,
                tango.io.protocol.model.IWriter;
+/
private import  tango.util.collection.model.HashParams,
                tango.util.collection.model.GuardIterator;

private import  tango.util.collection.impl.LLCell,
                tango.util.collection.impl.LLPair,
                tango.util.collection.impl.MapCollection,
                tango.util.collection.impl.AbstractIterator;

/*******************************************************************************

         Hash table implementation of Map
                
         author: Doug Lea
                @version 0.94

         <P> For an introduction to this package see <A HREF="index.html"
         > Overview </A>.

********************************************************************************/


public class HashMap(K, V) : MapCollection!(K, V), HashParams
{
        alias LLCell!(V)                LLCellT;
        alias LLPair!(K, V)             LLPairT;

        alias MapCollection!(K, V).remove     remove;
        alias MapCollection!(K, V).removeAll  removeAll;

        // instance variables

        /***********************************************************************

                The table. Each entry is a list. Null if no table allocated

        ************************************************************************/
  
        private LLPairT table[];

        /***********************************************************************

                The threshold load factor

        ************************************************************************/

        private float loadFactor;


        // constructors

        /***********************************************************************

                Make a new empty map to use given element screener.
        
        ************************************************************************/

        public this (Predicate screener = null)
        {
                this(screener, defaultLoadFactor);
        }

        /***********************************************************************

                Special version of constructor needed by clone()
        
        ************************************************************************/

        protected this (Predicate s, float f)
        {
                super(s);
                table = null;
                loadFactor = f;
        }

        /***********************************************************************

                Make an independent copy of the table. Elements themselves
                are not cloned.
        
        ************************************************************************/

        public final HashMap!(K, V) duplicate()
        {
                auto c = new HashMap (screener, loadFactor);

                if (count !is 0)
                   {
                   int cap = 2 * cast(int)((count / loadFactor)) + 1;
                   if (cap < defaultInitialBuckets)
                       cap = defaultInitialBuckets;

                   c.buckets(cap);

                   for (int i = 0; i < table.length; ++i)
                        for (LLPairT p = table[i]; p !is null; p = cast(LLPairT)(p.next()))
                             c.add (p.key(), p.element());
                   }
                return c;
        }


        // HashParams methods

        /***********************************************************************

                Implements tango.util.collection.HashParams.buckets.
                Time complexity: O(1).
                
                See_Also: tango.util.collection.HashParams.buckets.
        
        ************************************************************************/

        public final int buckets()
        {
                return (table is null) ? 0 : table.length;
        }

        /***********************************************************************

                Implements tango.util.collection.HashParams.buckets.
                Time complexity: O(n).
                
                See_Also: tango.util.collection.HashParams.buckets.
        
        ************************************************************************/

        public final void buckets(int newCap)
        {
                if (newCap is buckets())
                    return ;
                else
                   if (newCap >= 1)
                       resize(newCap);
                   else
                      throw new IllegalArgumentException("Invalid Hash table size");
        }

        /***********************************************************************

                Implements tango.util.collection.HashParams.thresholdLoadfactor
                Time complexity: O(1).
                
                See_Also: tango.util.collection.HashParams.thresholdLoadfactor
        
        ************************************************************************/

        public final float thresholdLoadFactor()
        {
                return loadFactor;
        }

        /***********************************************************************

                Implements tango.util.collection.HashParams.thresholdLoadfactor
                Time complexity: O(n).
                
                See_Also: tango.util.collection.HashParams.thresholdLoadfactor
        
        ************************************************************************/

        public final void thresholdLoadFactor(float desired)
        {
                if (desired > 0.0)
                   {
                   loadFactor = desired;
                   checkLoadFactor();
                   }
                else
                   throw new IllegalArgumentException("Invalid Hash table load factor");
        }



        // View methods

        /***********************************************************************

                Implements tango.util.collection.model.View.View.contains.
                Time complexity: O(1) average; O(n) worst.
                
                See_Also: tango.util.collection.model.View.View.contains
        
        ************************************************************************/
        
        public final bool contains(V element)
        {
                if (!isValidArg(element) || count is 0)
                    return false;

                for (int i = 0; i < table.length; ++i)
                    {
                    LLPairT hd = table[i];
                    if (hd !is null && hd.find(element) !is null)
                        return true;
                    }
                return false;
        }

        /***********************************************************************

                Implements tango.util.collection.model.View.View.instances.
                Time complexity: O(n).
                
                See_Also: tango.util.collection.model.View.View.instances
        
        ************************************************************************/
        
        public final uint instances(V element)
        {
                if (!isValidArg(element) || count is 0)
                    return 0;
    
                uint c = 0;
                for (uint i = 0; i < table.length; ++i)
                    {
                    LLPairT hd = table[i];
                    if (hd !is null)
                        c += hd.count(element);
                    }
                return c;
        }

        /***********************************************************************

                Implements tango.util.collection.model.View.View.elements.
                Time complexity: O(1).
                
                See_Also: tango.util.collection.model.View.View.elements
        
        ************************************************************************/
        
        public final GuardIterator!(V) elements()
        {
                return keys();
        }

        /***********************************************************************

                Implements tango.util.collection.model.View.View.opApply
                Time complexity: O(n)
                
                See_Also: tango.util.collection.model.View.View.opApply
        
        ************************************************************************/
        
        int opApply (int delegate (inout V value) dg)
        {
                auto scope iterator = new MapIterator!(K, V)(this);
                return iterator.opApply (dg);
        }


        /***********************************************************************

                Implements tango.util.collection.MapView.opApply
                Time complexity: O(n)
                
                See_Also: tango.util.collection.MapView.opApply
        
        ************************************************************************/
        
        int opApply (int delegate (inout K key, inout V value) dg)
        {
                auto scope iterator = new MapIterator!(K, V)(this);
                return iterator.opApply (dg);
        }


        // Map methods

        /***********************************************************************

                Implements tango.util.collection.Map.containsKey.
                Time complexity: O(1) average; O(n) worst.
                
                See_Also: tango.util.collection.Map.containsKey
        
        ************************************************************************/
        
        public final bool containsKey(K key)
        {
                if (!isValidKey(key) || count is 0)
                    return false;

                LLPairT p = table[hashOf(key)];
                if (p !is null)
                    return p.findKey(key) !is null;
                else
                   return false;
        }

        /***********************************************************************

                Implements tango.util.collection.Map.containsPair
                Time complexity: O(1) average; O(n) worst.
                
                See_Also: tango.util.collection.Map.containsPair
        
        ************************************************************************/
        
        public final bool containsPair(K key, V element)
        {
                if (!isValidKey(key) || !isValidArg(element) || count is 0)
                    return false;

                LLPairT p = table[hashOf(key)];
                if (p !is null)
                    return p.find(key, element) !is null;
                else
                   return false;
        }

        /***********************************************************************

                Implements tango.util.collection.Map.keys.
                Time complexity: O(1).
                
                See_Also: tango.util.collection.Map.keys
        
        ************************************************************************/
        
        public final PairIterator!(K, V) keys()
        {
                return new MapIterator!(K, V)(this);
        }

        /***********************************************************************

                Implements tango.util.collection.Map.get.
                Time complexity: O(1) average; O(n) worst.
                
                See_Also: tango.util.collection.Map.at
        
        ************************************************************************/
        
        public final V get(K key)
        {
                checkKey(key);
                if (count !is 0)
                   {
                   LLPairT p = table[hashOf(key)];
                   if (p !is null)
                      {
                      LLPairT c = p.findKey(key);
                      if (c !is null)
                          return c.element();
                      }
                   }
                throw new NoSuchElementException("no matching key");
        }


        /***********************************************************************

                Return the element associated with Key key. 
                @param key a key
                Returns: whether the key is contained or not
        
        ************************************************************************/

        public bool get(K key, inout V element)
        {
                checkKey(key);
                if (count !is 0)
                   {
                   LLPairT p = table[hashOf(key)];
                   if (p !is null)
                      {
                      LLPairT c = p.findKey(key);
                      if (c !is null)
                         {
                         element = c.element();
                         return true;
                         }
                      }
                   }
                return false;
        }



        /***********************************************************************

                Implements tango.util.collection.Map.keyOf.
                Time complexity: O(n).
                
                See_Also: tango.util.collection.Map.akyOf
        
        ************************************************************************/
        
        public final bool keyOf(inout K key, V value)
        {
                if (!isValidArg(value) || count is 0)
                    return false;

                for (int i = 0; i < table.length; ++i)
                    { 
                    LLPairT hd = table[i];
                    if (hd !is null)
                       {
                       auto p = (cast(LLPairT)(hd.find(value)));
                       if (p !is null)
                          {
                          key = p.key();
                          return true;
                          }
                       }
                    }
                return false;
        }


        // Collection methods

        /***********************************************************************

                Implements tango.util.collection.impl.Collection.Collection.clear.
                Time complexity: O(1).
                
                See_Also: tango.util.collection.impl.Collection.Collection.clear
        
        ************************************************************************/
        
        public final void clear()
        {
                setCount(0);
                table = null;
        }

        /***********************************************************************

                Implements tango.util.collection.impl.Collection.Collection.removeAll.
                Time complexity: O(n).
                
                See_Also: tango.util.collection.impl.Collection.Collection.removeAll
        
        ************************************************************************/
        
        public final void removeAll (V element)
        {
                remove_(element, true);
        }


        /***********************************************************************

                Implements tango.util.collection.impl.Collection.Collection.removeOneOf.
                Time complexity: O(n).
                
                See_Also: tango.util.collection.impl.Collection.Collection.removeOneOf
        
        ************************************************************************/
        
        public final void remove (V element)
        {
                remove_(element, false);
        }


        /***********************************************************************

                Implements tango.util.collection.impl.Collection.Collection.replaceOneOf.
                Time complexity: O(n).
                
                See_Also: tango.util.collection.impl.Collection.Collection.replaceOneOf
        
        ************************************************************************/

        public final void replace (V oldElement, V newElement)
        {
                replace_(oldElement, newElement, false);
        }

        /***********************************************************************

                Implements tango.util.collection.impl.Collection.Collection.replaceOneOf.
                Time complexity: O(n).
                
                See_Also: tango.util.collection.impl.Collection.Collection.replaceOneOf
        
        ************************************************************************/

        public final void replaceAll (V oldElement, V newElement)
        {
                replace_(oldElement, newElement, true);
        }

        /***********************************************************************

                Implements tango.util.collection.impl.Collection.Collection.take.
                Time complexity: O(number of buckets).
                
                See_Also: tango.util.collection.impl.Collection.Collection.take
        
        ************************************************************************/
        
        public final V take()
        {
                if (count !is 0)
                   {
                   for (int i = 0; i < table.length; ++i)
                       {
                       if (table[i] !is null)
                          {
                          decCount();
                          auto v = table[i].element();
                          table[i] = cast(LLPairT)(table[i].next());
                          return v;
                          }
                       }
                   }
                checkIndex(0);
                return V.init; // not reached
        }

        // Map methods

        /***********************************************************************

                Implements tango.util.collection.Map.add.
                Time complexity: O(1) average; O(n) worst.
                
                See_Also: tango.util.collection.Map.add
        
        ************************************************************************/
        
        public final void add (K key, V element)
        {
                checkKey(key);
                checkElement(element);

                if (table is null)
                    resize (defaultInitialBuckets);

                int h = hashOf(key);
                LLPairT hd = table[h];
                if (hd is null)
                   {
                   table[h] = new LLPairT(key, element, hd);
                   incCount();
                   return;
                   }
                else
                   {
                   LLPairT p = hd.findKey(key);
                   if (p !is null)
                      {
                      if (p.element() != (element))
                         {
                         p.element(element);
                         incVersion();
                         }
                      }
                   else
                      {
                      table[h] = new LLPairT(key, element, hd);
                      incCount();
                      checkLoadFactor(); // we only check load factor on add to nonempty bin
                      }
                   }
        }


        /***********************************************************************

                Implements tango.util.collection.Map.remove.
                Time complexity: O(1) average; O(n) worst.
                
                See_Also: tango.util.collection.Map.remove
        
        ************************************************************************/
        
        public final void removeKey (K key)
        {
                if (!isValidKey(key) || count is 0)
                    return;

                int h = hashOf(key);
                LLPairT hd = table[h];
                LLPairT p = hd;
                LLPairT trail = p;

                while (p !is null)
                      {
                      LLPairT n = cast(LLPairT)(p.next());
                      if (p.key() == (key))
                         {
                         decCount();
                         if (p is hd)
                             table[h] = n;
                         else
                            trail.unlinkNext();
                         return;
                         }
                      else
                         {
                         trail = p;
                         p = n;
                         }
                      }
        }

        /***********************************************************************

                Implements tango.util.collection.Map.replaceElement.
                Time complexity: O(1) average; O(n) worst.
                
                See_Also: tango.util.collection.Map.replaceElement
        
        ************************************************************************/
        
        public final void replacePair (K key, V oldElement, V newElement)
        {
                if (!isValidKey(key) || !isValidArg(oldElement) || count is 0)
                    return;

                LLPairT p = table[hashOf(key)];
                if (p !is null)
                   {
                   LLPairT c = p.find(key, oldElement);
                   if (c !is null)
                      {
                      checkElement(newElement);
                      c.element(newElement);
                      incVersion();
                      }
                   }
        }

        // Helper methods

        /***********************************************************************

                Check to see if we are past load factor threshold. If so,
                resize so that we are at half of the desired threshold.
                Also while at it, check to see if we are empty so can just
                unlink table.
        
        ************************************************************************/
        
        protected final void checkLoadFactor()
        {
                if (table is null)
                   {
                   if (count !is 0)
                       resize(defaultInitialBuckets);
                   }
                else
                   {
                   float fc = cast(float) (count);
                   float ft = table.length;

                   if (fc / ft > loadFactor)
                      {
                      int newCap = 2 * cast(int)(fc / loadFactor) + 1;
                      resize(newCap);
                      }
                   }
        }

        /***********************************************************************

                Mask off and remainder the hashCode for element
                so it can be used as table index
        
        ************************************************************************/

        protected final int hashOf(K key)
        {
                return (typeid(K).getHash(&key) & 0x7FFFFFFF) % table.length;
        }


        /***********************************************************************

        ************************************************************************/

        protected final void resize(int newCap)
        {
                LLPairT newtab[] = new LLPairT[newCap];

                if (table !is null)
                   {
                   for (int i = 0; i < table.length; ++i)
                       {
                       LLPairT p = table[i];
                       while (p !is null)
                             {
                             LLPairT n = cast(LLPairT)(p.next());
                             int h = (p.keyHash() & 0x7FFFFFFF) % newCap;
                             p.next(newtab[h]);
                             newtab[h] = p;
                             p = n;
                             }
                       }
                   }
                table = newtab;
                incVersion();
        }

        // helpers

        /***********************************************************************

        ************************************************************************/

        private final void remove_(V element, bool allOccurrences)
        {
                if (!isValidArg(element) || count is 0)
                    return;

                for (int h = 0; h < table.length; ++h)
                    {
                    LLCellT hd = table[h];
                    LLCellT p = hd;
                    LLCellT trail = p;
                    while (p !is null)
                          {
                          LLPairT n = cast(LLPairT)(p.next());
                          if (p.element() == (element))
                             {
                             decCount();
                             if (p is table[h])
                                {
                                table[h] = n;
                                trail = n;
                                }
                             else
                                trail.next(n);
                             if (! allOccurrences)
                                   return;
                             else
                                p = n;
                             }
                          else
                             {
                             trail = p;
                             p = n;
                             }
                          }
                    }
        }

        /***********************************************************************

        ************************************************************************/

        private final void replace_(V oldElement, V newElement, bool allOccurrences)
        {
                if (count is 0 || !isValidArg(oldElement) || oldElement == (newElement))
                    return;

                for (int h = 0; h < table.length; ++h)
                    {
                    LLCellT hd = table[h];
                    LLCellT p = hd;
                    LLCellT trail = p;
                    while (p !is null)
                          {
                          LLPairT n = cast(LLPairT)(p.next());
                          if (p.element() == (oldElement))
                             {
                             checkElement(newElement);
                             incVersion();
                             p.element(newElement);
                             if (! allOccurrences)
                                   return ;
                             }
                          trail = p;
                          p = n;
                          }
                    }
        }

/+
        // IReader & IWriter methods

        /***********************************************************************

        ************************************************************************/

        public override void read (IReader input)
        {
                int     len;
                K       key;
                V       element;
                
                input (len) (loadFactor) (count);
                table = (len > 0) ? new LLPairT[len] : null;

                for (len=count; len-- > 0;)
                    {
                    input (key) (element);
                    
                    int h = hashOf (key);
                    table[h] = new LLPairT (key, element, table[h]);
                    }
        }
                        
        /***********************************************************************

        ************************************************************************/

        public override void write (IWriter output)
        {
                output (table.length) (loadFactor) (count);

                if (table.length > 0)
                    foreach (key, value; keys)
                             output (key) (value);
        }
        
+/
        // ImplementationCheckable methods

        /***********************************************************************

                Implements tango.util.collection.model.View.View.checkImplementation.
                
                See_Also: tango.util.collection.model.View.View.checkImplementation
        
        ************************************************************************/
                        
        public override void checkImplementation()
        {
                super.checkImplementation();

                assert(!(table is null && count !is 0));
                assert((table is null || table.length > 0));
                assert(loadFactor > 0.0f);

                if (table is null)
                    return;

                int c = 0;
                for (int i = 0; i < table.length; ++i)
                    {
                    for (LLPairT p = table[i]; p !is null; p = cast(LLPairT)(p.next()))
                        {
                        ++c;
                        assert(allows(p.element()));
                        assert(allowsKey(p.key()));
                        assert(containsKey(p.key()));
                        assert(contains(p.element()));
                        assert(instances(p.element()) >= 1);
                        assert(containsPair(p.key(), p.element()));
                        assert(hashOf(p.key()) is i);
                        }
                    }
                assert(c is count);


        }


        /***********************************************************************

                opApply() has migrated here to mitigate the virtual call
                on method get()
                
        ************************************************************************/

        private static class MapIterator(K, V) : AbstractMapIterator!(K, V)
        {
                private int             row;
                private LLPairT         pair;
                private LLPairT[]       table;

                public this (HashMap map)
                {
                        super (map);
                        table = map.table;
                }

                public final V get(inout K key)
                {
                        auto v = get();
                        key = pair.key;
                        return v;
                }

                public final V get()
                {
                        decRemaining();

                        if (pair)
                            pair = cast(LLPairT) pair.next();

                        while (pair is null)
                               pair = table [row++];

                        return pair.element();
                }

                int opApply (int delegate (inout V value) dg)
                {
                        int result;

                        for (auto i=remaining(); i--;)
                            {
                            auto value = get();
                            if ((result = dg(value)) != 0)
                                 break;
                            }
                        return result;
                }

                int opApply (int delegate (inout K key, inout V value) dg)
                {
                        K   key;
                        int result;

                        for (auto i=remaining(); i--;)
                            {
                            auto value = get(key);
                            if ((result = dg(key, value)) != 0)
                                 break;
                            }
                        return result;
                }
        }
}


debug(Test)
{
        import tango.io.Console;
                        
        void main()
        {
                auto map = new HashMap!(char[], double);
                map.add ("foo", 3.14);
                map.add ("bar", 6.28);

                foreach (key, value; map.keys) {typeof(key) x; x = key;}

                foreach (value; map.keys) {}

                foreach (value; map.elements) {}

                auto keys = map.keys();
                while (keys.more)
                       auto v = keys.get();

                foreach (value; map) {}

                foreach (key, value; map)
                         Cout (key).newline;

                map.checkImplementation();

                Cout (map).newline;
        }
}