Browse Source

完成跳表的插入以及压测一千万次,10240个随机key值48s

Gogs 4 months ago
parent
commit
180afa3205
4 changed files with 158 additions and 94 deletions
  1. 5 1
      .gitignore
  2. 107 77
      STL/extend/skiplist.h
  3. 27 16
      test.cpp
  4. 19 0
      tests/time.h

+ 5 - 1
.gitignore

@@ -28,9 +28,13 @@
 *.out
 *.app
 
-
+#buid dir
 build/
+
+#about vs
 .vscode/
+
+#Compiled result of test
 test
 
 

+ 107 - 77
STL/extend/skiplist.h

@@ -1,13 +1,15 @@
-#include"basical/type_checking.h"
+#include"../basical/type_checking.h"
 #include<cstdlib>
 #include<cstring> 
 #include<exception>
 #include<random>
 #include<shared_mutex>
 #include <variant>
+#include<chrono>
 
 namespace nanxing_extend
 {
+    static int count=0;
     //错误处理机制
     class nextpoint_new:std::exception             //skip_node分配空间失败的时候
     {
@@ -51,21 +53,22 @@ namespace nanxing_extend
     template<typename K,typename V>
     struct skip_node
     {
-        static_assert(nanxing_extend::compare_admit<K>value);
-        static_assert(nanxing_extend::compare_admit<V>value);
+        static_assert(nanxing_extend::compare_admit<K>::value,"the type of K is error");
+        static_assert(nanxing_extend::compare_admit<V>::value,"the type of V is error");
         skip_node<K,V>** next_node; 
         V value;
-    private:
         K key;
+    private:
+        
         int level;
     public:
         skip_node(){};
         skip_node(K _key,V _value,int _level):key(_key),value(_value),level(_level){};
-        init_next(int level)
+        void init_next(int level)
         {
             try
             {
-                next_node=::new (skip_node<K,V>*)[level];
+                next_node=::new skip_node<K,V>*[level];
             }
             catch(std::bad_alloc){        //捕获内存分配错误
                 throw nextpoint_new();       //重新抛出一个定制的更详细的类型用以明确错误的具体位置
@@ -78,19 +81,20 @@ namespace nanxing_extend
     template<typename K,typename V>
     class skipList
     {
-        static_assert(nanxing_extend::compare_admit<K>value);
-        static_assert(nanxing_extend::compare_admit<V>value);
+        static_assert(nanxing_extend::compare_admit<K>::value,"the type of K is error");
+        static_assert(nanxing_extend::compare_admit<V>::value,"the type of V is error");
     private:
         using Node=skip_node<K,V>;
         using ptr=Node*;
         using Nptr=Node**;
-        //由于C++的便利性我们考虑使用带头节点的跳表C++允许对数据不进行初始化(默认构造函数))
+        //由于C++的便利性我们考虑使用带头节点的跳表(C++允许对数据不进行初始化(默认构造函数))
     #ifdef NANXING_THREAD_
         std::shared_mutex RW_lock;           //读写锁
     #endif
         Nptr head;                      //头节点
         int max_level;                 //最大高度
-        int* random_level;
+        int* random_level;             //如果启用随机数表这个就非空,反之为nullptr
+        //当不启用随机数表,使用rand()构造随机数,启用的时候用mt19773构造随机数
         int current_level;              //跳表当前高度
         int current_size;               //跳表当前尺寸
         //这里出于一个考虑,当跳表单纯作为小数据内存数据库,单表大小限制是没有意义的
@@ -99,25 +103,30 @@ namespace nanxing_extend
         int max_size;                   //跳表允许的最大尺寸
     #endif
     public:
-        #ifndef SKIP_MAX_SIZE
+
+    #ifndef SKIP_MAX_SIZE
         skipList(int _max_level):max_level(_max_level),random_level(nullptr)
         {
             try
             {
-                Node* middle=::new skip_node;
+                Node* middle=::new skip_node<K,V>;
                 middle->init_next(max_level);
-                head=::new (Node*)[max_level];
-                for(auto& i in head)                
+                head=::new  Node*[max_level];
+                for(int i=0;i<max_level;i++)                
                 {
-                    i=middle;
+                    head[i]=middle;
                 }
             }
             catch(std::bad_alloc)
             {
                 throw newNode_error();        //重新抛出更详细的错误类型
             }
+            if(max_level==0){                //如果将高度设置为0直接调用terminate打断整个程序执行
+                std::cerr<<"the level of skiplist cannot set zero"<<std::endl;
+                std::terminate();
+            }
         }
-        #elif
+    #elif
         skipList(int _max_level,int _max_size):max_size(_max_size),max_level(_max_level),random_level(nullptr)
         {
             try
@@ -135,7 +144,7 @@ namespace nanxing_extend
                 throw newNode_error();
             }
         }
-        #endif
+    #endif
 
         [[nodiscard]]                            //这个返回值最好不要忽略否则很有可能会出现内存泄漏
         auto insert(K _key,V _value)->std::variant<Skip_result,V>              //如果相同的时候我们考虑将value返回因为value很可能会是一个指针,需要手动清空内存
@@ -143,7 +152,7 @@ namespace nanxing_extend
         #ifdef NANXING_THREAD_
             std::lock_guard<std::shared_mutex> lock(RW_lock);             
         #endif
-        std::variant<Skip_result,V> sk;
+        
         #ifdef SKIP_MAX_SIZE
             if(current_size==max_size)
             {
@@ -152,76 +161,74 @@ namespace nanxing_extend
         #endif
             int rand_level=0;
             ptr updata[max_level]={nullptr};          //用于更新的数组
-            int level=current_level-1;
-            ptr point=head[level];
+            ptr point=head[max_level-1];
             
             ptr new_node;
-            for(int i=level;i>=0;i--)
+            std::variant<Skip_result,V> sk;
+            for(int i=max_level-1;i>=0;i--)
             {
                 for(;;)
                 {
-                    if(point->next_node[level]==nullptr)
+                    if(point->next_node[i]==nullptr)
                     {
-                        updata[level]=point;
+                        updata[i]=point;
                         break;
                     }
-                    else if(point->next_node[level]->key>=_key)
+                    else if(point->next_node[i]->key>=_key)
                     {
-                        if(point->next_node[level]->key==_key)
+                        if(point->next_node[i]->key==_key)
                         {
-                            sk=std::move(point->next_node[level]->value);          //这个值已经不需要了,直接移动
-                            point->next_node[level]->value=_value;
+                            sk=std::move(point->next_node[i]->value);          //这个值已经不需要了,直接移动
+                            point->next_node[i]->value=_value;
                             return sk;
                         }
                         else
                         {
-                            updata[level]=point;
+                            updata[i]=point;
                             break;
                         }
-                        else{
-                            point=point->head[level];              //更新point指针
-                        }
+                    }
+                    else{
+                        point=point->next_node[i];              //更新point指针
                     }
                 }
-                [[likely]]
-                if(random_level!=nullptr)
-                {
-                    rand_level=random_level[current_size%1024];
-                    new_node=new skip_node(_key,_value,rand_level);
-                }
-                [[unlikely]]
-                else
-                {
-                    rand_level=std::rand(chrono::system_clock::now().time_since_epoch().count())%(1+max_level);
-                    new_node=new skip_node(_key,_value,rand_level);
-                }
-                ptr tmp=nullptr;
-                
-                for(int i=0;i<=rand_level;i++)
-                {
-                    tmp=updata[i]->next_node[i];
-                    updata[i]->next_node[i]=new_node;
-                    new_node->next_node[i]=tmp;
-                }
-                if(rand_level>current_level)
-                {
-                    current_level=rand_level;
-                }
-                current_size++;
+            }                
+            [[likely]]
+            if(random_level!=nullptr)
+            {
+                rand_level=random_level[current_size%1024];
+            }
+            else
+            {
+                rand_level=rand()%max_level;
+            }
+            ptr tmp=nullptr;
+            new_node=new skip_node(_key,_value,rand_level);
+            new_node->init_next(rand_level);
+            for(int i=0;i<rand_level;i++)
+            {
+                tmp=updata[i]->next_node[i];
+                updata[i]->next_node[i]=new_node;
+                new_node->next_node[i]=tmp;
+            }
+            if(rand_level>current_level)
+            {
+                current_level=rand_level;
             }
+            current_size++;
             sk=Skip_result::successufl;
+            return sk;
         }
 
         [[nodiscard]]                    
-        auto search(K _key)->std::variant<Skip_result,V>  noexcept       //不涉及任何内存分配相关任务,因此是异常安全的
-        {
+        auto search(K _key) noexcept ->std::variant<Skip_result,V>{       //不涉及任何内存分配相关任务,因此是异常安全的
         #ifdef NANXING_THREAD_
             std::shared_lock<std::shared_mutex> lock(RW_lock);             
         #endif        
             std::variant<Skip_result,V> sk;
             ptr tmp=head[current_level-1];
             int tmp_level=current_level-1;
-            for(i=tmp_level;i>=0;i--)
+            for(int i=tmp_level;i>=0;i--)
             {
                 while(tmp->next_node[tmp_level]!=nullptr)
                 {
@@ -253,39 +260,62 @@ namespace nanxing_extend
                 return;
             }            
             try{
-                random_level=new int[1024];                     //刚好是一页的大小(4KB)
+                random_level=::new int[1024];                     //刚好是一页的大小(4KB)
             }
             catch(std::bad_alloc)
             {
                 throw random_error();
             }
-            std::mt19937 rnd(chrono::system_clock::now().time_since_epoch().count());
-            for(auto& i :random_level)
+            std::mt19937 rnd(std::chrono::system_clock::now().time_since_epoch().count());
+            for(int i=0;i<1024;i++)
             {
-                i=(rnd()%max_level)+1;
+                random_level[i]=(rnd()%max_level)+1;
             }
-
         }
-    };
 
-    #ifdef SKIP_MAX_SIZE
-    [[nodicard]]
-        auto change_size(int _max_size)->std::variant<Skip_result,V> noexcept
+
+        void Print()
         {
-            std::variant<
-            if(_max_size>this->max_size)
+            ptr tmp=head[0]->next_node[0];
+            while(tmp!=nullptr&&tmp->next_node[0]!=nullptr)         //这里用了截断的技巧,即第一个条件不成立就不会触发第二个条件运行
             {
-                this->max_size=_max_size;
-                tmp=Skip_result::successufl;
-                return tmp;
+                std::cout<<"("<<tmp->get_key()<<","<<tmp->value<<")"<<"->";
+                tmp=tmp->next_node[0];
+                count++;
             }
-            else
+            if(tmp!=nullptr)
             {
-                tmp=Skip_result::too_samll;
-                return tmp;
+                std::cout<<"("<<tmp->get_key()<<","<<tmp->value<<")"<<std::endl;
+                count++;
             }
-        } 
+            std::cout<<"count ="<<count<<std::endl;
+        }
+    
+
+    #ifdef SKIP_MAX_SIZE
+        [[nodicard]]
+            auto change_size(int _max_size)->std::variant<Skip_result,V> noexcept
+            {
+                std::variant<
+                if(_max_size>this->max_size)
+                {
+                    this->max_size=_max_size;
+                    tmp=Skip_result::successufl;
+                    return tmp;
+                }
+                else
+                {
+                    tmp=Skip_result::too_samll;
+                    return tmp;
+                }
+            } 
     #endif
+
+    
+    
+    };
+
+
 }
 
 

+ 27 - 16
test.cpp

@@ -1,28 +1,39 @@
 #include<iostream>
 #include<typeinfo>
-template<typename TYPE>
-union ty
+#include"STL/extend/skiplist.h"
+#include<functional>
+#include<future>
+#include<chrono>
+#include<random>
+int fun(int a,int b)
 {
-    int k,
-    TYPE type
-};
+    return a+b;
+}
 
 int main()
 {
-    ty<double> t;
-    t.k=15;
-    if(typeid(t)==typeid(double))
+    nanxing_extend::skipList<int,int> test{15};
+    int random[1024];
+    std::mt19937 rnd(std::chrono::system_clock::now().time_since_epoch().count());
+    for(auto& i:random)
     {
-        std::cout<<"int"<<t.TYPE<<std::endl;
+        i=static_cast<unsigned int>(rnd());
     }
-    int a[10]={0};
-    for(auto& i : a)
+    std::chrono::high_resolution_clock::time_point tp1 = std::chrono::high_resolution_clock::now();
+    
+    for(int i=0;i<200000;i++)
     {
-        i=1;
-    }
-    for(auto j :a)
-    {
-        std::cout<<j<<std::endl;
+        auto get=test.insert(random[i%1024],i);
     }
 
+    std::chrono::high_resolution_clock::time_point tp2 = std::chrono::high_resolution_clock::now();
+    std::chrono::duration<size_t, std::nano> dur = tp2 - tp1;
+    test.Print();
+    std::cout << "1 被计时代码耗时:" << std::chrono::duration_cast<std::chrono::nanoseconds>(dur).count() << " 纳秒" << std::endl;
+    std::cout << "2 被计时代码耗时:" << std::chrono::duration_cast<std::chrono::microseconds>(dur).count() << " 微妙" << std::endl;
+    std::cout << "3 被计时代码耗时:" << std::chrono::duration_cast<std::chrono::milliseconds>(dur).count() << " 毫秒" << std::endl;
+    std::cout << "4 被计时代码耗时:" << std::chrono::duration_cast<std::chrono::seconds>(dur).count() << " 秒钟" << std::endl;
+    std::cout << "5 被计时代码耗时:" << std::chrono::duration_cast<std::chrono::minutes>(dur).count() << " 分钟" << std::endl;
+    std::cout << "6 被计时代码耗时:" << std::chrono::duration_cast<std::chrono::hours>(dur).count() << " 小时" << std::endl;
+
 }

+ 19 - 0
tests/time.h

@@ -0,0 +1,19 @@
+#include<chrono>
+#include<functional>
+#include<iostream>
+namespace nanxing_test
+{
+    void time_test(std::function<void()> fun )
+    {
+        std::chrono::high_resolution_clock::time_point tp1 = std::chrono::high_resolution_clock::now();
+        fun();
+        std::chrono::high_resolution_clock::time_point tp2 = std::chrono::high_resolution_clock::now();
+        std::chrono::duration<size_t, std::nano> dur = tp2 - tp1;
+        std::cout << "1 被计时代码耗时:" << std::chrono::duration_cast<std::chrono::nanoseconds>(dur).count() << " 纳秒" << std::endl;
+        std::cout << "2 被计时代码耗时:" << std::chrono::duration_cast<std::chrono::microseconds>(dur).count() << " 微妙" << std::endl;
+        std::cout << "3 被计时代码耗时:" << std::chrono::duration_cast<std::chrono::milliseconds>(dur).count() << " 毫秒" << std::endl;
+        std::cout << "4 被计时代码耗时:" << std::chrono::duration_cast<std::chrono::seconds>(dur).count() << " 秒钟" << std::endl;
+        std::cout << "5 被计时代码耗时:" << std::chrono::duration_cast<std::chrono::minutes>(dur).count() << " 分钟" << std::endl;
+        std::cout << "6 被计时代码耗时:" << std::chrono::duration_cast<std::chrono::hours>(dur).count() << " 小时" << std::endl;
+    }
+}