Blame view
src/util/hash-list.h
5.55 KB
8dcb6dfcb first commit |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 |
// util/hash-list.h // Copyright 2009-2011 Microsoft Corporation // 2013 Johns Hopkins University (author: Daniel Povey) // See ../../COPYING for clarification regarding multiple authors // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // THIS CODE IS PROVIDED *AS IS* BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY // KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION ANY IMPLIED // WARRANTIES OR CONDITIONS OF TITLE, FITNESS FOR A PARTICULAR PURPOSE, // MERCHANTABLITY OR NON-INFRINGEMENT. // See the Apache 2 License for the specific language governing permissions and // limitations under the License. #ifndef KALDI_UTIL_HASH_LIST_H_ #define KALDI_UTIL_HASH_LIST_H_ #include <vector> #include <set> #include <algorithm> #include <limits> #include <cassert> #include "util/stl-utils.h" /* This header provides utilities for a structure that's used in a decoder (but is quite generic in nature so we implement and test it separately). Basically it's a singly-linked list, but implemented in such a way that we can quickly search for elements in the list. We give it a slightly richer interface than just a hash and a list. The idea is that we want to separate the hash part and the list part: basically, in the decoder, we want to have a single hash for the current frame and the next frame, because by the time we need to access the hash for the next frame we no longer need the hash for the previous frame. So we have an operation that clears the hash but leaves the list structure intact. We also control memory management inside this object, to avoid repeated new's/deletes. See hash-list-test.cc for an example of how to use this object. */ namespace kaldi { template<class I, class T> class HashList { public: struct Elem { I key; T val; Elem *tail; }; /// Constructor takes no arguments. /// Call SetSize to inform it of the likely size. HashList(); /// Clears the hash and gives the head of the current list to the user; /// ownership is transferred to the user (the user must call Delete() /// for each element in the list, at his/her leisure). Elem *Clear(); /// Gives the head of the current list to the user. Ownership retained in the /// class. Caution: in December 2013 the return type was changed to const /// Elem* and this function was made const. You may need to change some types /// of local Elem* variables to const if this produces compilation errors. const Elem *GetList() const; /// Think of this like delete(). It is to be called for each Elem in turn /// after you "obtained ownership" by doing Clear(). This is not the opposite /// of. Insert, it is the opposite of New. It's really a memory operation. inline void Delete(Elem *e); /// This should probably not be needed to be called directly by the user. /// Think of it as opposite /// to Delete(); inline Elem *New(); /// Find tries to find this element in the current list using the hashtable. /// It returns NULL if not present. The Elem it returns is not owned by the /// user, it is part of the internal list owned by this object, but the user /// is free to modify the "val" element. inline Elem *Find(I key); /// Insert inserts a new element into the hashtable/stored list. /// Because element keys in a hashtable are unique, this operation checks /// whether each inserted element has a key equivalent to the one of an /// element already in the hashtable. If so, the element is not inserted, /// returning an pointer to this existing element. inline Elem *Insert(I key, T val); /// Insert inserts another element with same key into the hashtable/ /// stored list. /// By calling this, the user asserts that one element with that key is /// already present. /// We insert it that way, that all elements with the same key /// follow each other. /// Find() will return the first one of the elements with the same key. inline void InsertMore(I key, T val); /// SetSize tells the object how many hash buckets to allocate (should /// typically be at least twice the number of objects we expect to go in the /// structure, for fastest performance). It must be called while the hash /// is empty (e.g. after Clear() or after initializing the object, but before /// adding anything to the hash. void SetSize(size_t sz); /// Returns current number of hash buckets. inline size_t Size() { return hash_size_; } ~HashList(); private: struct HashBucket { size_t prev_bucket; // index to next bucket (-1 if list tail). Note: // list of buckets goes in opposite direction to list of Elems. Elem *last_elem; // pointer to last element in this bucket (NULL if empty) inline HashBucket(size_t i, Elem *e): prev_bucket(i), last_elem(e) {} }; Elem *list_head_; // head of currently stored list. size_t bucket_list_tail_; // tail of list of active hash buckets. size_t hash_size_; // number of hash buckets. std::vector<HashBucket> buckets_; Elem *freed_head_; // head of list of currently freed elements. [ready for // allocation] std::vector<Elem*> allocated_; // list of allocated blocks. static const size_t allocate_block_size_ = 1024; // Number of Elements to // allocate in one block. Must be largish so storing allocated_ doesn't // become a problem. }; } // end namespace kaldi #include "util/hash-list-inl.h" #endif // KALDI_UTIL_HASH_LIST_H_ |