zoukankan      html  css  js  c++  java
  • hashtable

    参考《stl源码剖析》,用开链法实现了简易hash表,并进行了简单测试。

    特点:

    1. hash表可扩容。当元素个数大于hash表长的时候,hash表挑选下一个质数,扩容一倍。

    2. 提供了hash函数以及equal函数的载入

    3. 提供find,insert,erase三个基本操作。

        find返回某一元素的指针

        insert可选择不重复插入或可重复插入

        erase可选择删除一个元素或所有元素

    4. 省略了迭代器,静态数组未封装在类内

      1 #include <bits/stdc++.h>
      2 using namespace std;
      3 
      4 namespace ht {
      5     const  int _num_prime = 22;
      6     static const int prime_list[_num_prime] = {
      7         53,         97,           193,         389,       769,
      8         1543,       3079,         6151,        12289,     24593,
      9         49157,      98317,        196613,      393241,    786433,
     10         1572869,    3145739,      6291469,     12582917,  25165843,
     11         50331653,   100663319 
     12     };
     13 
     14     template<class Value>
     15     struct hashtable_node {
     16         Value val;
     17         hashtable_node *next;
     18         hashtable_node(Value val, hashtable_node *next):val(val), next(next){}
     19     };
     20     /*
     21     template<class Key>
     22     struct hashtable_iterator {
     23         typedef hashtable_node<Key> node;
     24         node *cur;
     25     };
     26     */
     27     template<class Key, class Hash = std::hash<Key>, class KeyEqual = std::equal_to<Key>>
     28     struct hashtable{
     29         typedef hashtable_node<Key> node;
     30         
     31         Hash hash;
     32         KeyEqual equals;
     33         
     34         vector<node*> buckets;
     35         int num_elements;
     36 
     37         hashtable(int n = *prime_list){
     38             num_elements = 0;
     39             int n_buckets = next_size(n);
     40             buckets.resize(n_buckets);
     41         }
     42         ~hashtable(){
     43             for(auto cur: buckets) {
     44                 while(cur != NULL) {
     45                     node *tmp = cur->next;
     46                     delete cur;
     47                     cur = tmp;
     48                 }
     49             }
     50         }
     51 
     52         node* find(const Key& obj) {
     53             int n = hash(obj)%buckets.size();
     54             node *cur = buckets[n];
     55             while(cur) { 
     56                 if (equals(cur->val, obj)) 
     57                     return cur; 
     58                 cur = cur->next;
     59             }
     60             return cur;
     61         }    
     62         
     63         pair<node*, bool> insert(const Key& obj, bool same = false) {
     64             resize(num_elements + 1);
     65             const int n = hash(obj)%buckets.size();
     66             node *first = buckets[n], *tmp = NULL;
     67             if(same || (tmp = find(obj)) == NULL) {
     68                 tmp = new node(obj, first);
     69                 buckets[n] = tmp;
     70                 ++num_elements;
     71                 return {tmp, true};
     72             }
     73             return {tmp, false};
     74         }
     75 
     76         bool erase(const Key& obj, bool same = false) {
     77             if(find(obj) == NULL) return false;        
     78             int n = hash(obj)%buckets.size();
     79             node **cur = &buckets[n];
     80             while(*cur) {
     81                 if(equals((**cur).val, obj)) {
     82                     node *tmp = *cur;
     83                     *cur = tmp->next;
     84                     delete tmp;
     85                     --num_elements;
     86                     if(!same) break ;
     87                 }
     88                 else cur = &(**cur).next;
     89             }        
     90             return true;
     91         }
     92         
     93         void debug() {
     94             puts("debug...");
     95             bool tag = false;
     96             for(auto cur: buckets) {
     97                 tag = false;
     98                 while(cur) {
     99                     tag = true;
    100                     printf("%d ", cur->val);
    101                     cur = cur->next;
    102                 }
    103                 if(tag) puts("");
    104             }
    105             puts("end debug...");
    106         }
    107 
    108     private:
    109         int next_size(int n) {
    110             const int *last = prime_list+_num_prime;
    111             const int *pos = lower_bound(prime_list, last, n);
    112             return pos == last? *(last-1) : *pos;
    113         }
    114         void resize(int num) {
    115             int old_n = buckets.size();
    116             if(num > old_n){
    117                 int n = next_size(num);
    118                 if(n > old_n){
    119                     vector<node*> tmp(n, NULL);
    120                     for(int bucket = 0; bucket < old_n; ++bucket){
    121                         node *first = buckets[bucket];
    122                         while(first){
    123                             int new_bucket = hash(first->val)%n;
    124                             buckets[bucket] = first->next;
    125                             first->next = tmp[new_bucket];
    126                             tmp[new_bucket] = first;
    127                             first = buckets[bucket];
    128                         }
    129                     }
    130                     buckets.swap(tmp);
    131                 }
    132             }
    133         }
    134     };
    135 }
    136 
    137 int main() {
    138     ht::hashtable<int> H;
    139     
    140     H.insert(3);
    141     H.insert(3, true);
    142     H.insert(3, true);
    143     H.insert(3, true);
    144     H.insert(3);
    145     H.debug();
    146     printf("%d
    ", H.erase(3));
    147     H.debug();
    148 
    149     auto p = H.find(3);
    150     printf("%d
    ",p? p->val: (int)p);
    151     printf("%d
    ", H.erase(3, true));
    152 
    153     p = H.find(3);
    154     printf("%d
    ",p? p->val: (int)p); 
    155 
    156     H.insert(3);
    157     H.insert(3, true);
    158     H.insert(3, true);
    159     H.insert(3);
    160 
    161     H.insert(5);
    162     H.insert(5);
    163     H.debug();
    164     p = H.find(5);
    165     printf("%d
    ",p? p->val: (int)p);
    166 
    167     printf("%d
    ", H.erase(5));
    168     printf("%d
    ", H.erase(3));
    169     p = H.find(5);
    170     printf("%d
    ",p? p->val: (int)p);
    171     H.debug();
    172     return 0;
    173 }
  • 相关阅读:
    下拉框插件开发 (一)
    工厂模式 抽象工厂模式
    闭包 构建函数 简单工厂模式
    js继承
    浅谈js观察者模式
    浅谈js单例模式
    localstorage本地存储
    Vi 配置文件
    Vi Command
    我的Firs博客
  • 原文地址:https://www.cnblogs.com/dirge/p/9607722.html
Copyright © 2011-2022 走看看