zoukankan      html  css  js  c++  java
  • redis的数据结构——跳跃表

    一、什么是跳跃表

    首先先思考一个这个问题,如果给定一个有序数组[1, 3, 5, 7, 9, 11, 13, 15],那么我们怎么能快速的查找到一个元素呢?

    我们可以用二分法进行查找,假如我们要查找的是13,首先取数组中间的值7,由于中间元素7小于13,那么接着再取右边的中间元素11,而11小于13,继续取右边的中间元素13,命中。如果长度为n,二分查找的时间复杂度为O(logn)相信大家都能很轻易的想出这个问题。

    那么再思考一个问题,如果给定的是一个有序链表呢?1-3-5-7-9-11-13-15,我们怎么才能快速获取到其中的值呢?

    显然二分查找是不可行的。这个时候就需要用到跳跃表这种数据结构。

    二、跳跃表的实现原理

    我们知道普通单项链表的查询时间复杂度是O(n),即使单项链表是有序的,我们也不能通过二分的方式缩减时间复杂度。

    如果要查询节点6,那么我们需要从节点1开始一个一个查找,需要比较六次才能得到结果,那么怎么才能减少比较次数呢,我们可以将链表中的节点提出一些来再建立一个链表,就像书的目录一样

     这是如果我们想要查询6,首先在第一层查找四次没有查询到,因为节点是有序的,那么我们从节点5处到达第一层进行查找,只需一次查询即可,因此共需要5次查询。目前节点比较少,查询差别不大,当节点变多时,差异效果就出来了。当然,我们还能够增加跳跃表的层数来获取更快的查询效果。

     这是一个三层的跳跃表,这个时候如果我们想要查询系节点7,那么,我们只需要比较三次就够了,大大提高查询效率。当然,我们还能再构建一层

     这是查询7只需要比较两次就够了,查询6时最耗时的,需要比较5次,跳跃表的思想和二分法是非常相似的。那么跳跃表的时间复杂度时多少呢?

    如果有n个元素,因为是2分,所以层数就应该是log n层 (本文所有log都是以2为底),再加上自身的1层。以上图为例,如果是4个元素,那么分层为L3和L4,再加上本身的L2,一共3层;如果是8个元素,那么就是3+1层。最耗时间的查询自然是访问所有层数,耗时logn+logn,即2logn。为什么是2倍的logn呢?我们以上图中的46为例,查询到46要访问所有的分层,每个分层都要访问2个元素,中间元素和最后一个元素。所以时间复杂度为O(logn)。

    至此为止,我们引入了最理想的跳跃表,但是如果想要在上图中插入或者删除一个元素呢?比如我们要插入一个元素8,自然在L1层,我们将这些元素插入在元素8后,那么L2层,L3层呢?我们是不是要考虑插入后怎样调整连接,才能维持这个理想的跳跃表结构。我们知道,平衡二叉树的调整是一件令人头痛的事情,左旋右旋左右旋……一般人还真记不住,而调整一个理想的跳跃表将是一个比调整平衡二叉树还复杂的操作。幸运的是,我们并不需要通过复杂的操作调整连接来维护这样完美的跳跃表。有一种基于概率统计的插入算法,也能得到时间复杂度为O(logn)的查询效率,这种跳跃表才是我们真正要实现的。

    三、容易实现的跳跃表

    先讨论插入,我们先看理想的跳跃表结构,L2层的元素个数是L1层元素个数的1/2,L3层的元素个数是L2层的元素个数的1/2,以此类推。从这里,我们可以想到,只要在插入时尽量保证上一层的元素个数是下一层元素的1/2,我们的跳跃表就能成为理想的跳跃表。那么怎么样才能在插入时保证上一层元素个数是下一层元素个数的1/2呢?很简单,抛硬币就能解决了!假设元素X要插入跳跃表,很显然,L1层肯定要插入X。那么L2层要不要插入X呢?我们希望上层元素个数是下层元素个数的1/2,所以我们有1/2的概率希望X插入L2层,那么抛一下硬币吧,正面就插入,反面就不插入。那么L3到底要不要插入X呢?相对于L2层,我们还是希望1/2的概率插入,那么继续抛硬币吧!以此类推,元素X插入第n层的概率是(1/2)的n次。这样,我们能在跳跃表中插入一个元素了。

    以下图为例:跳跃表的初试状态如下图,表中没有一个元素:

     接下来插入元素2,首先肯定要在第一层插入这个元素

     那么第二次要不要插入这个元素呢,我们通过抛硬币,结果是正面,要插入。如图

    继续抛硬币,结果是反面,那么元素2的插入操作就停止了。 接下来,我们插入元素33,跟元素2的插入一样,现在L1层插入33,如下图:

     然后抛硬币,结果是反面,那么元素33的插入操作就结束了,插入后的表结构就是上图所示。接下来,我们插入元素55,首先在L1插入55,插入后如下图:

     然后抛硬币,结果是正面,那么L2层需要插入55,如下图:

     继续抛硬币,结果又是正面,那么L3层需要插入55,如下图:

     继续抛硬币,结果又是正面,那么要在L4插入55,结果如下图:

     

    继续抛硬币,结果是反面,那么55的插入结束,表结构就如上图所示。

    以此类推,我们插入剩余的元素。当然因为规模小,结果很可能不是一个理想的跳跃表。但是如果元素个数n的规模很大,学过概率论的同学都知道,最终的表结构肯定非常接近于理想跳跃表。

    当然,这样的分析在感性上是很直接的,但是时间复杂度的证明实在复杂,在此我就不深究了,感兴趣的可以去看关于跳跃表的paper。

    再讨论删除,删除操作没什么讲的,直接删除元素,然后调整一下删除元素后的指针即可。跟普通的链表删除操作完全一样。

    再来讨论一下时间复杂度,插入和删除的时间复杂度就是查询元素插入位置的时间复杂度,这不难理解,所以是O(logn)。

    四、java实现

    在章节2中,我们采用抛硬币的方式来决定新元素插入的最高层数,这当然不能在程序中实现。代码中,我们采用随机数生成的方式来获取新元素插入的最高层数。我们先估摸一下n的规模,然后定义跳跃表的最大层数maxLevel,那么底层,也就是第0层,元素是一定要插入的,概率为1;最高层,也就是maxLevel层,元素插入的概率为1/2^maxLevel。

    我们先随机生成一个范围为0~2^maxLevel-1的一个整数r。那么元素r小于2^(maxLevel-1)的概率为1/2,r小于2^(maxLevel-2)的概率为1/4,……,r小于2的概率为1/2^(maxLevel-1),r小于1的概率为1/2^maxLevel。

    举例,假设maxLevel为4,那么r的范围为0~15,则r小于8的概率为1/2,r小于4的概率为1/4,r小于2的概率为1/8,r小于1的概率为1/16。1/16正好是maxLevel层插入元素的概率,1/8正好是maxLevel层插入的概率,以此类推。

    通过这样的分析,我们可以先比较r和1,如果r<1,那么元素就要插入到maxLevel层以下;否则再比较r和2,如果r<2,那么元素就要插入到maxLevel-1层以下;再比较r和4,如果r<4,那么元素就要插入到maxLevel-2层以下……如果r>2^(maxLevel - 1),那么元素就只要插入在底层即可。

    以上分析是随机数算法的关键。算法跟实现跟语言无关,但是Java程序员还是更容易看明白Java代码实现的跳跃表,以下贴一下别人的java代码实现。作者找不到了,就这样吧。

     

    /***************************  SkipList.java  *********************/
    
    import java.util.Random;
    
    public class SkipList<T extends Comparable<? super T>> {
        private int maxLevel;
        private SkipListNode<T>[] root;
        private int[] powers;
        private Random rd = new Random();
        SkipList() {
            this(4);
        }
        SkipList(int i) {
            maxLevel = i;
            root = new SkipListNode[maxLevel];
            powers = new int[maxLevel];
            for (int j = 0; j < maxLevel; j++)
                root[j] = null;
            choosePowers();
        }
        public boolean isEmpty() {
            return root[0] == null;
        }
        public void choosePowers() {
            powers[maxLevel-1] = (2 << (maxLevel-1)) - 1;    // 2^maxLevel - 1
            for (int i = maxLevel - 2, j = 0; i >= 0; i--, j++)
               powers[i] = powers[i+1] - (2 << j);           // 2^(j+1)
        }
        public int chooseLevel() {
            int i, r = Math.abs(rd.nextInt()) % powers[maxLevel-1] + 1;
            for (i = 1; i < maxLevel; i++)
                if (r < powers[i])
                    return i-1; // return a level < the highest level;
            return i-1;         // return the highest level;
        }
        // make sure (with isEmpty()) that search() is called for a nonempty list;
        public T search(T key) {
            int lvl;
            SkipListNode<T> prev, curr;            // find the highest nonnull
            for (lvl = maxLevel-1; lvl >= 0 && root[lvl] == null; lvl--); // level;
            prev = curr = root[lvl];
            while (true) {
                if (key.equals(curr.key))          // success if equal;
                     return curr.key;
                else if (key.compareTo(curr.key) < 0) { // if smaller, go down,
                     if (lvl == 0)                 // if possible
                          return null;
                     else if (curr == root[lvl])   // by one level
                          curr = root[--lvl];      // starting from the
                     else curr = prev.next[--lvl]; // predecessor which
                }                                  // can be the root;
                else {                             // if greater,
                     prev = curr;                  // go to the next
                     if (curr.next[lvl] != null)   // non-null node
                          curr = curr.next[lvl];   // on the same level
                     else {                        // or to a list on a lower level;
                          for (lvl--; lvl >= 0 && curr.next[lvl] == null; lvl--);
                          if (lvl >= 0)
                               curr = curr.next[lvl];
                          else return null;
                     }
                }
            }
        }
        public void insert(T key) {
            SkipListNode<T>[] curr = new SkipListNode[maxLevel];
            SkipListNode<T>[] prev = new SkipListNode[maxLevel];
            SkipListNode<T> newNode;
            int lvl, i;
            curr[maxLevel-1] = root[maxLevel-1];
            prev[maxLevel-1] = null;
            for (lvl = maxLevel - 1; lvl >= 0; lvl--) {
                while (curr[lvl] != null && curr[lvl].key.compareTo(key) < 0) {
                    prev[lvl] = curr[lvl];           // go to the next
                    curr[lvl] = curr[lvl].next[lvl]; // if smaller;
                }
                if (curr[lvl] != null && key.equals(curr[lvl].key)) // don't
                    return;                          // include duplicates;
                if (lvl > 0)                         // go one level down
                    if (prev[lvl] == null) {         // if not the lowest
                          curr[lvl-1] = root[lvl-1]; // level, using a link
                          prev[lvl-1] = null;        // either from the root
                    }
                    else {                           // or from the predecessor;
                         curr[lvl-1] = prev[lvl].next[lvl-1];
                         prev[lvl-1] = prev[lvl];
                    }
            }
            lvl = chooseLevel();                // generate randomly level
            newNode = new SkipListNode<T>(key,lvl+1); // for newNode;
            for (i = 0; i <= lvl; i++) {        // initialize next fields of
                newNode.next[i] = curr[i];      // newNode and reset to newNode
                if (prev[i] == null)            // either fields of the root
                     root[i] = newNode;         // or next fields of newNode's
                else prev[i].next[i] = newNode; // predecessors;
            }
        }
    }

     

  • 相关阅读:
    @雅礼集训01/13
    @hdu
    @bzoj
    @hdu
    @bzoj
    @雅礼集训01/10
    @codeforces
    @spoj
    @bzoj
    @bzoj
  • 原文地址:https://www.cnblogs.com/ChenBingJie123/p/13557211.html
Copyright © 2011-2022 走看看