1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxHashTable class 
   4 // Author:      Julian Smart 
   5 // Modified by: VZ at 25.02.00: type safe hashes with WX_DECLARE_HASH() 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  15 #if defined(__GNUG__) && !defined(__APPLE__) 
  16 #pragma interface "hash.h" 
  20 #include "wx/dynarray.h" 
  22 // the default size of the hash 
  23 #define wxHASH_SIZE_DEFAULT     (1000) 
  26  * A hash table is an array of user-definable size with lists 
  27  * of data items hanging off the array positions.  Usually there'll 
  28  * be a hit, so no search is required; otherwise we'll have to run down 
  29  * the list to find the desired item. 
  32 // ---------------------------------------------------------------------------- 
  33 // this is the base class for object hashes: hash tables which contain 
  34 // pointers to objects 
  35 // ---------------------------------------------------------------------------- 
  37 class WXDLLEXPORT wxHashTableBase 
: public wxObject
 
  42     void Create(wxKeyType keyType 
= wxKEY_INTEGER
, 
  43                 size_t size 
= wxHASH_SIZE_DEFAULT
); 
  46     size_t GetSize() const { return m_hashSize
; } 
  47     size_t GetCount() const { return m_count
; } 
  49     void DeleteContents(bool flag
); 
  52     // find the node for (key, value) 
  53     wxNodeBase 
*GetNode(long key
, long value
) const; 
  55     // the array of lists in which we store the values for given key hash 
  56     wxListBase 
**m_hashTable
; 
  58     // the size of m_lists array 
  61     // the type of indexing we use 
  64     // the total number of elements in the hash 
  67     // should we delete our data? 
  68     bool m_deleteContents
; 
  71     // no copy ctor/assignment operator (yet) 
  72     DECLARE_NO_COPY_CLASS(wxHashTableBase
) 
  75 // ---------------------------------------------------------------------------- 
  76 // a hash table which stores longs 
  77 // ---------------------------------------------------------------------------- 
  79 class WXDLLEXPORT wxHashTableLong 
: public wxObject
 
  82     wxHashTableLong(size_t size 
= wxHASH_SIZE_DEFAULT
) 
  84     virtual ~wxHashTableLong(); 
  86     void Create(size_t size 
= wxHASH_SIZE_DEFAULT
); 
  89     size_t GetSize() const { return m_hashSize
; } 
  90     size_t GetCount() const { return m_count
; } 
  92     void Put(long key
, long value
); 
  93     long Get(long key
) const; 
  94     long Delete(long key
); 
  97     void Init(size_t size
); 
 100     wxArrayLong 
**m_values
, 
 103     // the size of array above 
 106     // the total number of elements in the hash 
 109     // not implemented yet 
 110     DECLARE_NO_COPY_CLASS(wxHashTableLong
) 
 113 // ---------------------------------------------------------------------------- 
 114 // wxStringHashTable: a hash table which indexes strings with longs 
 115 // ---------------------------------------------------------------------------- 
 117 class WXDLLEXPORT wxStringHashTable 
: public wxObject
 
 120     wxStringHashTable(size_t sizeTable 
= wxHASH_SIZE_DEFAULT
); 
 121     virtual ~wxStringHashTable(); 
 123     // add a string associated with this key to the table 
 124     void Put(long key
, const wxString
& value
); 
 126     // get the string from the key: if not found, an empty string is returned 
 127     // and the wasFound is set to FALSE if not NULL 
 128     wxString 
Get(long key
, bool *wasFound 
= NULL
) const; 
 130     // remove the item, returning TRUE if the item was found and deleted 
 131     bool Delete(long key
) const; 
 137     wxArrayLong 
**m_keys
; 
 138     wxArrayString 
**m_values
; 
 140     // the size of array above 
 143     DECLARE_NO_COPY_CLASS(wxStringHashTable
) 
 146 // ---------------------------------------------------------------------------- 
 147 // for compatibility only 
 148 // ---------------------------------------------------------------------------- 
 150 class WXDLLEXPORT wxHashTable 
: public wxObject
 
 154     int current_position
; 
 155     wxNode 
*current_node
; 
 157     unsigned int key_type
; 
 160     wxHashTable(int the_key_type 
= wxKEY_INTEGER
, 
 161                 int size 
= wxHASH_SIZE_DEFAULT
); 
 164     // copy ctor and assignment operator 
 165     wxHashTable(const wxHashTable
& table
) : wxObject() 
 167     wxHashTable
& operator=(const wxHashTable
& table
) 
 168         { Clear(); DoCopy(table
); return *this; } 
 170     void DoCopy(const wxHashTable
& table
); 
 174     bool Create(int the_key_type 
= wxKEY_INTEGER
, 
 175                 int size 
= wxHASH_SIZE_DEFAULT
); 
 177     // Note that there are 2 forms of Put, Get. 
 178     // With a key and a value, the *value* will be checked 
 179     // when a collision is detected. Otherwise, if there are 
 180     // 2 items with a different value but the same key, 
 181     // we'll retrieve the WRONG ONE. So where possible, 
 182     // supply the required value along with the key. 
 183     // In fact, the value-only versions make a key, and still store 
 184     // the value. The use of an explicit key might be required 
 185     // e.g. when combining several values into one key. 
 186     // When doing that, it's highly likely we'll get a collision, 
 187     // e.g. 1 + 2 = 3, 2 + 1 = 3. 
 189     // key and value are NOT necessarily the same 
 190     void Put(long key
, long value
, wxObject 
*object
); 
 191     void Put(long key
, const wxChar 
*value
, wxObject 
*object
); 
 193     // key and value are the same 
 194     void Put(long value
, wxObject 
*object
); 
 195     void Put(const wxChar 
*value
, wxObject 
*object
); 
 197     // key and value not the same 
 198     wxObject 
*Get(long key
, long value
) const; 
 199     wxObject 
*Get(long key
, const wxChar 
*value
) const; 
 201     // key and value are the same 
 202     wxObject 
*Get(long value
) const; 
 203     wxObject 
*Get(const wxChar 
*value
) const; 
 205     // Deletes entry and returns data if found 
 206     wxObject 
*Delete(long key
); 
 207     wxObject 
*Delete(const wxChar 
*key
); 
 209     wxObject 
*Delete(long key
, int value
); 
 210     wxObject 
*Delete(long key
, const wxChar 
*value
); 
 212     // Construct your own integer key from a string, e.g. in case 
 213     // you need to combine it with something 
 214     long MakeKey(const wxChar 
*string
) const; 
 216     // Way of iterating through whole hash table (e.g. to delete everything) 
 217     // Not necessary, of course, if you're only storing pointers to 
 218     // objects maintained separately 
 223     void DeleteContents(bool flag
); 
 226     // Returns number of nodes 
 227     size_t GetCount() const { return m_count
; } 
 230     size_t m_count
;             // number of elements in the hashtable 
 231     bool m_deleteContents
; 
 233     DECLARE_DYNAMIC_CLASS(wxHashTable
) 
 236 // defines a new type safe hash table which stores the elements of type eltype 
 237 // in lists of class listclass 
 238 #define _WX_DECLARE_HASH(eltype, listclass, hashclass, classexp)               \ 
 239     classexp hashclass : public wxHashTableBase                                \ 
 242         hashclass(wxKeyType keyType = wxKEY_INTEGER,                           \ 
 243                   size_t size = wxHASH_SIZE_DEFAULT)                           \ 
 244             { Create(keyType, size); }                                         \ 
 246         ~hashclass() { Destroy(); }                                            \ 
 248         void Put(long key, long val, eltype *data) { DoPut(key, val, data); }  \ 
 249         void Put(long key, eltype *data) { DoPut(key, key, data); }            \ 
 251         eltype *Get(long key, long value) const                                \ 
 253             wxNodeBase *node = GetNode(key, value);                            \ 
 254             return node ? ((listclass::Node *)node)->GetData() : (eltype *)0;  \ 
 256         eltype *Get(long key) const { return Get(key, key); }                  \ 
 258         eltype *Delete(long key, long value)                                   \ 
 262             wxNodeBase *node = GetNode(key, value);                            \ 
 265                 data = ((listclass::Node *)node)->GetData();                   \ 
 272                 data = (eltype *)0;                                            \ 
 277         eltype *Delete(long key) { return Delete(key, key); }                  \ 
 280         void DoPut(long key, long value, eltype *data)                         \ 
 282             size_t slot = (size_t)abs((int)(key % (long)m_hashSize));          \ 
 284             if ( !m_hashTable[slot] )                                          \ 
 286                 m_hashTable[slot] = new listclass(m_keyType);                  \ 
 287                 if ( m_deleteContents )                                        \ 
 288                     m_hashTable[slot]->DeleteContents(TRUE);                   \ 
 291             ((listclass *)m_hashTable[slot])->Append(value, data);             \ 
 296 // this macro is to be used in the user code 
 297 #define WX_DECLARE_HASH(el, list, hash) \ 
 298     _WX_DECLARE_HASH(el, list, hash, class) 
 300 // and this one does exactly the same thing but should be used inside the 
 302 #define WX_DECLARE_EXPORTED_HASH(el, list, hash)  \ 
 303     _WX_DECLARE_HASH(el, list, hash, class WXDLLEXPORT) 
 305 #define WX_DECLARE_USER_EXPORTED_HASH(el, list, hash, usergoo)  \ 
 306     _WX_DECLARE_HASH(el, list, hash, class usergoo)