leveldb 學習記錄(四) skiplist補與變長數字

leveldb 學習記錄(一) skiplist 已經將skiplist的插入 查找等操做流程用圖示說明html

這裏在介紹 下skiplist的代碼node

裏面有幾個模塊  算法

template<typename Key, class Comparator>
class SkipList {......}數組

class Arena;(內存池模塊 暫時不介紹)app

struct Node;(節點 存儲key 和指向其餘Node的指針)dom

 

//Node 構造函數  KEY賦值
// Implementation details follow
template<typename Key, class Comparator>
struct SkipList<Key,Comparator>::Node {
  explicit Node(const Key& k) : key(k) { }

  Key const key;

  // Accessors/mutators for links.  Wrapped in methods so we can
  // add the appropriate barriers as necessary.
  // 訪問修改器使用 包裝在方法中 能夠使用內存屏障
  // Node指向的另外一層?
  Node* Next(int n) {
    assert(n >= 0);
    // Use an 'acquire load' so that we observe a fully initialized
    // version of the returned Node.
    return reinterpret_cast<Node*>(next_[n].Acquire_Load());
  }
  //設置下層Node指向
  void SetNext(int n, Node* x) {
    assert(n >= 0);
    // Use a 'release store' so that anybody who reads through this
    // pointer observes a fully initialized version of the inserted node.
    next_[n].Release_Store(x);
  }

  // No-barrier variants that can be safely used in a few locations.
  // 讀取本Node在N層的NODE指向  非內存屏蔽操做
  Node* NoBarrier_Next(int n) {
    assert(n >= 0);
    return reinterpret_cast<Node*>(next_[n].NoBarrier_Load());
  }
  
  //設置Node 在N層的Node指向  非內存屏蔽操做 
  void NoBarrier_SetNext(int n, Node* x) {
    assert(n >= 0);
    next_[n].NoBarrier_Store(x);
  }

 private:
  // Array of length equal to the node height.  next_[0] is lowest level link.
  // 原子指針數組  指向其餘Node  建立時候動態確認數組長度
  port::AtomicPointer next_[1];
};

  

結構示意圖以下ide

其餘操做可參見系列文章一函數

最後附上我註釋的代碼oop

  1 #include <assert.h>
  2 #include <stdlib.h>
  3 #include "port/port.h"
  4 #include "util/arena.h"
  5 #include "util/random.h"
  6 
  7 namespace leveldb {
  8 
  9 class Arena;
 10 
 11 template<typename Key, class Comparator>
 12 class SkipList {
 13  private:
 14   struct Node;
 15 
 16  public:
 17   // Create a new SkipList object that will use "cmp" for comparing keys,
 18   // and will allocate memory using "*arena".  Objects allocated in the arena
 19   // must remain allocated for the lifetime of the skiplist object.
 20   explicit SkipList(Comparator cmp, Arena* arena);
 21 
 22   // Insert key into the list.
 23   // REQUIRES: nothing that compares equal to key is currently in the list.
 24   void Insert(const Key& key);
 25 
 26   // Returns true iff an entry that compares equal to key is in the list.
 27   bool Contains(const Key& key) const;
 28 
 29   // Iteration over the contents of a skip list
 30   class Iterator {
 31    public:
 32     // Initialize an iterator over the specified list.
 33     // The returned iterator is not valid.
 34     explicit Iterator(const SkipList* list);
 35 
 36     // Returns true iff the iterator is positioned at a valid node.
 37     bool Valid() const;
 38 
 39     // Returns the key at the current position.
 40     // REQUIRES: Valid()
 41     const Key& key() const;
 42 
 43     // Advances to the next position.
 44     // REQUIRES: Valid()
 45     void Next();
 46 
 47     // Advances to the previous position.
 48     // REQUIRES: Valid()
 49     void Prev();
 50 
 51     // Advance to the first entry with a key >= target
 52     void Seek(const Key& target);
 53 
 54     // Position at the first entry in list.
 55     // Final state of iterator is Valid() iff list is not empty.
 56     void SeekToFirst();
 57 
 58     // Position at the last entry in list.
 59     // Final state of iterator is Valid() iff list is not empty.
 60     void SeekToLast();
 61 
 62    private:
 63     const SkipList* list_;
 64     Node* node_;
 65     // Intentionally copyable
 66   };
 67 
 68  private:
 69   enum { kMaxHeight = 12 };
 70 
 71   // Immutable after construction
 72   Comparator const compare_;
 73   Arena* const arena_;    // Arena used for allocations of nodes
 74 
 75   Node* const head_;
 76 
 77   // Modified only by Insert().  Read racily by readers, but stale
 78   // values are ok.
 79   port::AtomicPointer max_height_;   // Height of the entire list
 80 
 81   inline int GetMaxHeight() const {
 82     return reinterpret_cast<intptr_t>(max_height_.NoBarrier_Load());
 83   }
 84 
 85   // Read/written only by Insert().
 86   Random rnd_;
 87 
 88   Node* NewNode(const Key& key, int height);
 89   int RandomHeight();
 90   bool Equal(const Key& a, const Key& b) const { return (compare_(a, b) == 0); }
 91 
 92   // Return true if key is greater than the data stored in "n"
 93   bool KeyIsAfterNode(const Key& key, Node* n) const;
 94 
 95   // Return the earliest node that comes at or after key.
 96   // Return NULL if there is no such node.
 97   //
 98   // If prev is non-NULL, fills prev[level] with pointer to previous
 99   // node at "level" for every level in [0..max_height_-1].
100   Node* FindGreaterOrEqual(const Key& key, Node** prev) const;
101 
102   // Return the latest node with a key < key.
103   // Return head_ if there is no such node.
104   Node* FindLessThan(const Key& key) const;
105 
106   // Return the last node in the list.
107   // Return head_ if list is empty.
108   Node* FindLast() const;
109 
110   // No copying allowed
111   SkipList(const SkipList&);
112   void operator=(const SkipList&);
113 };
114 
115 
116 //Node 構造函數  KEY賦值
117 // Implementation details follow
118 template<typename Key, class Comparator>
119 struct SkipList<Key,Comparator>::Node {
120   explicit Node(const Key& k) : key(k) { }
121 
122   Key const key;
123 
124   // Accessors/mutators for links.  Wrapped in methods so we can
125   // add the appropriate barriers as necessary.
126   // 訪問修改器使用 包裝在方法中 能夠使用內存屏障
127   // Node指向的另外一層?
128   Node* Next(int n) {
129     assert(n >= 0);
130     // Use an 'acquire load' so that we observe a fully initialized
131     // version of the returned Node.
132     return reinterpret_cast<Node*>(next_[n].Acquire_Load());
133   }
134   //設置下層Node指向
135   void SetNext(int n, Node* x) {
136     assert(n >= 0);
137     // Use a 'release store' so that anybody who reads through this
138     // pointer observes a fully initialized version of the inserted node.
139     next_[n].Release_Store(x);
140   }
141 
142   // No-barrier variants that can be safely used in a few locations.
143   // 讀取本Node在N層的NODE指向  非內存屏蔽操做
144   Node* NoBarrier_Next(int n) {
145     assert(n >= 0);
146     return reinterpret_cast<Node*>(next_[n].NoBarrier_Load());
147   }
148   
149   //設置Node 在N層的Node指向  非內存屏蔽操做 
150   void NoBarrier_SetNext(int n, Node* x) {
151     assert(n >= 0);
152     next_[n].NoBarrier_Store(x);
153   }
154 
155  private:
156   // Array of length equal to the node height.  next_[0] is lowest level link.
157   // 原子指針數組  指向其餘Node  建立時候動態確認數組長度
158   port::AtomicPointer next_[1];
159 };
160 
161 
162 //建立一個NODE 內存池arena_分配內存  height確認該NODE的Node指向數量
163 template<typename Key, class Comparator>
164 typename SkipList<Key,Comparator>::Node*
165 SkipList<Key,Comparator>::NewNode(const Key& key, int height) {
166   char* mem = arena_->AllocateAligned(
167       sizeof(Node) + sizeof(port::AtomicPointer) * (height - 1));
168   return new (mem) Node(key);
169 }
170 
171 //根據輸入的SkipList指針構造一個  iterator
172 template<typename Key, class Comparator>
173 inline SkipList<Key,Comparator>::Iterator::Iterator(const SkipList* list) {
174   list_ = list;
175   node_ = NULL;
176 }
177 
178 //
179 template<typename Key, class Comparator>
180 inline bool SkipList<Key,Comparator>::Iterator::Valid() const {
181   return node_ != NULL;
182 }
183 
184 
185 //返回iterator指向的Node的key
186 template<typename Key, class Comparator>
187 inline const Key& SkipList<Key,Comparator>::Iterator::key() const {
188   assert(Valid());
189   return node_->key;
190 }
191 
192 
193 //獲取iterator指向的下一個node  在第0層獲取(Node第0層均有記錄)
194 template<typename Key, class Comparator>
195 inline void SkipList<Key,Comparator>::Iterator::Next() {
196   assert(Valid());
197   node_ = node_->Next(0);
198 }
199 
200 //尋找該node上一個node
201 template<typename Key, class Comparator>
202 inline void SkipList<Key,Comparator>::Iterator::Prev() {
203   // Instead of using explicit "prev" links, we just search for the
204   // last node that falls before key.
205   assert(Valid());
206   node_ = list_->FindLessThan(node_->key);
207   if (node_ == list_->head_) {
208     node_ = NULL;
209   }
210 }
211 
212 
213 //調用 FindGreaterOrEqual 查找 
214 template<typename Key, class Comparator>
215 inline void SkipList<Key,Comparator>::Iterator::Seek(const Key& target) {
216   node_ = list_->FindGreaterOrEqual(target, NULL);
217 }
218 
219 //重置到head  Node 第0層 第一個Node
220 template<typename Key, class Comparator>
221 inline void SkipList<Key,Comparator>::Iterator::SeekToFirst() {
222   node_ = list_->head_->Next(0);
223 }
224 
225 //重置到last Node
226 template<typename Key, class Comparator>
227 inline void SkipList<Key,Comparator>::Iterator::SeekToLast() {
228   node_ = list_->FindLast();
229   if (node_ == list_->head_) {
230     node_ = NULL;
231   }
232 }
233 
234 
235 //返回height  隨機增長1
236 template<typename Key, class Comparator>
237 int SkipList<Key,Comparator>::RandomHeight() {
238   // Increase height with probability 1 in kBranching
239   static const unsigned int kBranching = 4;
240   int height = 1;
241   while (height < kMaxHeight && ((rnd_.Next() % kBranching) == 0)) {
242     height++;
243   }
244   assert(height > 0);
245   assert(height <= kMaxHeight);
246   return height;
247 }
248 
249 //調用 cmp 比較是大小
250 template<typename Key, class Comparator>
251 bool SkipList<Key,Comparator>::KeyIsAfterNode(const Key& key, Node* n) const {
252   // NULL n is considered infinite
253   return (n != NULL) && (compare_(n->key, key) < 0);
254 }
255 
256 
257 //尋找比key大或者等於的Node
258 template<typename Key, class Comparator>
259 typename SkipList<Key,Comparator>::Node* SkipList<Key,Comparator>::FindGreaterOrEqual(const Key& key, Node** prev)
260     const {
261   Node* x = head_;
262   int level = GetMaxHeight() - 1;
263   //    k在Next後面 則在本level尋找 不然 在下一層level尋找
264   while (true) {
265     Node* next = x->(level);
266     if (KeyIsAfterNode(key, next)) {
267       // Keep searching in this list
268       x = next;
269     } else {
270       if (prev != NULL) prev[level] = x;
271       if (level == 0) {
272         return next;
273       } else {
274         // Switch to next list
275         level--;
276       }
277     }
278   }
279 }
280 
281 
282 //尋找小於等於KEY的第一個NODE
283 template<typename Key, class Comparator>
284 typename SkipList<Key,Comparator>::Node*
285 SkipList<Key,Comparator>::FindLessThan(const Key& key) const {
286   Node* x = head_;
287   int level = GetMaxHeight() - 1;
288   while (true) {
289     assert(x == head_ || compare_(x->key, key) < 0);
290     Node* next = x->Next(level);
291     if (next == NULL || compare_(next->key, key) >= 0) {
292       if (level == 0) {
293         return x;
294       } else {
295         // Switch to next list
296         level--;
297       }
298     } else {
299       x = next;
300     }
301   }
302 }
303 
304 //在最高層 尋找最後一個Node
305 template<typename Key, class Comparator>
306 typename SkipList<Key,Comparator>::Node* SkipList<Key,Comparator>::FindLast()
307     const {
308   Node* x = head_;
309   int level = GetMaxHeight() - 1;
310   while (true) {
311     Node* next = x->Next(level);
312     if (next == NULL) {
313       if (level == 0) {
314         return x;
315       } else {
316         // Switch to next list
317         level--;
318       }
319     } else {
320       x = next;
321     }
322   }
323 }
324 
325 
326 //構建函數 第一個head Node key是0 後繼Node指針最大kMaxHeight
327 template<typename Key, class Comparator>
328 SkipList<Key,Comparator>::SkipList(Comparator cmp, Arena* arena)
329     : compare_(cmp),
330       arena_(arena),
331       head_(NewNode(0 /* any key will do */, kMaxHeight)),
332       max_height_(reinterpret_cast<void*>(1)),
333       rnd_(0xdeadbeef) {
334   for (int i = 0; i < kMaxHeight; i++) {
335     head_->SetNext(i, NULL);
336   }
337 }
338 
339 //insert算法
340 template<typename Key, class Comparator>
341 void SkipList<Key,Comparator>::Insert(const Key& key) {
342   // TODO(opt): We can use a barrier-free variant of FindGreaterOrEqual()
343   // here since Insert() is externally synchronized.
344   Node* prev[kMaxHeight];
345   Node* x = FindGreaterOrEqual(key, prev);
346 
347   // Our data structure does not allow duplicate insertion
348   assert(x == NULL || !Equal(key, x->key));
349 
350   int height = RandomHeight();
351   if (height > GetMaxHeight()) {
352     for (int i = GetMaxHeight(); i < height; i++) {
353       prev[i] = head_;
354     }
355     //fprintf(stderr, "Change height from %d to %d\n", max_height_, height);
356 
357     // It is ok to mutate max_height_ without any synchronization
358     // with concurrent readers.  A concurrent reader that observes
359     // the new value of max_height_ will see either the old value of
360     // new level pointers from head_ (NULL), or a new value set in
361     // the loop below.  In the former case the reader will
362     // immediately drop to the next level since NULL sorts after all
363     // keys.  In the latter case the reader will use the new node.
364     max_height_.NoBarrier_Store(reinterpret_cast<void*>(height));
365   }
366 
367   x = NewNode(key, height);
368   for (int i = 0; i < height; i++) {
369     // NoBarrier_SetNext() suffices since we will add a barrier when
370     // we publish a pointer to "x" in prev[i].
371     x->NoBarrier_SetNext(i, prev[i]->NoBarrier_Next(i));
372     prev[i]->SetNext(i, x);
373   }
374 }
375 
376 
377 //尋找等於KEY 便可確認是否包含與KEY相等的Node
378 template<typename Key, class Comparator>
379 bool SkipList<Key,Comparator>::Contains(const Key& key) const {
380   Node* x = FindGreaterOrEqual(key, NULL);
381   if (x != NULL && Equal(key, x->key)) {
382     return true;
383   } else {
384     return false;
385   }
386 }
387 
388 }
View Code

 

 varint變長數post

 leveldb使用了Protocol Buffers的變長字節整形編碼 每一個字節中使用7位來表示數字 最高位用做他處

300的編碼,編碼後是兩個字節:

1010 1100 0000 0010

  它這個例子是每一個字節左邊是高位,能夠看到每一個字節的最高位是一個標識位,從左到右第一個字節是10101100,最高位是1,說明後面還有字節須要解碼,而後第二個字節是00000010,最高位是0,後面沒字節了。因此這兩個字節就須要解碼成一個整數,再往下看。

代碼
1010 1100 0000 0010
→ 010 1100 000 0010 //每一個字節去掉最高位
→ 000 0010 010 1100 //字節序轉換,兩個字節互換位置
→ 000 0010 ++ 010 1100 //兩個字節進行連接操做(不是相加)
→ 100101100 //合併後的結果,高位位0的部分截取掉
→ 256 + 32 + 8 + 4 = 300 //每一個值爲1的bit位乘以以2爲底的冪,得出編碼後的值

主要函數在 

coding.h

extern void PutFixed32(std::string* dst, uint32_t value);
extern void PutFixed64(std::string* dst, uint64_t value);
extern void PutVarint32(std::string* dst, uint32_t value);
extern void PutVarint64(std::string* dst, uint64_t value);
extern void PutLengthPrefixedSlice(std::string* dst, const Slice& value);

extern bool GetVarint32(Slice* input, uint32_t* value);
extern bool GetVarint64(Slice* input, uint64_t* value);
extern bool GetLengthPrefixedSlice(Slice* input, Slice* result);

extern const char* GetVarint32Ptr(const char* p,const char* limit, uint32_t* v);
extern const char* GetVarint64Ptr(const char* p,const char* limit, uint64_t* v);


extern int VarintLength(uint64_t v);

extern void EncodeFixed32(char* dst, uint32_t value);
extern void EncodeFixed64(char* dst, uint64_t value);

extern char* EncodeVarint32(char* dst, uint32_t value);
extern char* EncodeVarint64(char* dst, uint64_t value);

 

 

 

 

 

 

 

 

 

 

參考

https://www.cnblogs.com/onlytiancai/archive/2010/10/16/1853003.html

相關文章
相關標籤/搜索