zoukankan      html  css  js  c++  java
  • 跳表

    跳表:

     

     

     

    。。 

    代码实现:

    package com.mj;
    
    import java.util.Comparator;
    
    @SuppressWarnings("unchecked")
    public class SkipList<K, V> {
        private static final int MAX_LEVEL = 32;
        private static final double P = 0.25;
        private int size;
        private Comparator<K> comparator;
        /**
         * 有效层数
         */
        private int level;
        /**
         * 不存放任何K-V
         */
        private Node<K, V> first;
        
        public SkipList(Comparator<K> comparator) {
            this.comparator = comparator;
            first = new Node<>(null, null, MAX_LEVEL);
        }
        
        public SkipList() {
            this(null);
        }
        
        public int size() {
            return size;
        }
        
        public boolean isEmpty() {
            return size == 0;
        }
        
        public V get(K key) {
            keyCheck(key);
            
            // first.nexts[3] == 21节点
            // first.nexts[2] == 9节点
            // first.nexts[1] == 6节点
            // first.nexts[0] == 3节点
            
            // key = 30
            // level = 4
            
            Node<K, V> node = first;
            for (int i = level - 1; i >= 0; i--) {
                int cmp = -1;
                while (node.nexts[i] != null 
                        && (cmp = compare(key, node.nexts[i].key)) > 0) {
                    node = node.nexts[i];
                }
                // node.nexts[i].key >= key
                if (cmp == 0) return node.nexts[i].value;
            }
            return null;
        }
        
        public V put(K key, V value) {
            keyCheck(key);
            
            Node<K, V> node = first;
            Node<K, V>[] prevs = new Node[level];
            for (int i = level - 1; i >= 0; i--) {
                int cmp = -1;
                while (node.nexts[i] != null 
                        && (cmp = compare(key, node.nexts[i].key)) > 0) {
                    node = node.nexts[i];
                }
                if (cmp == 0) { // 节点是存在的
                    V oldV = node.nexts[i].value;
                    node.nexts[i].value = value;
                    return oldV;
                }
                prevs[i] = node;
            }
            
            // 新节点的层数
            int newLevel = randomLevel();
            // 添加新节点
            Node<K, V> newNode = new Node<>(key, value, newLevel);
            // 设置前驱和后继
            for (int i = 0; i < newLevel; i++) {
                if (i >= level) {
                    first.nexts[i] = newNode;
                } else {
                    newNode.nexts[i] = prevs[i].nexts[i];
                    prevs[i].nexts[i] = newNode;
                }
            }
            
            // 节点数量增加
            size++;
            
            // 计算跳表的最终层数
            level = Math.max(level, newLevel);
            
            return null;
        }
        
        public V remove(K key) {
            keyCheck(key);
            
            Node<K, V> node = first;
            Node<K, V>[] prevs = new Node[level];
            boolean exist = false;
            for (int i = level - 1; i >= 0; i--) {
                int cmp = -1;
                while (node.nexts[i] != null 
                        && (cmp = compare(key, node.nexts[i].key)) > 0) {
                    node = node.nexts[i];
                }
                prevs[i] = node;
                if (cmp == 0) exist = true;
            }
            if (!exist) return null;
            
            // 需要被删除的节点
            Node<K, V> removedNode = node.nexts[0];
            
            // 数量减少
            size--;
            
            // 设置后继
            for (int i = 0; i < removedNode.nexts.length; i++) {
                prevs[i].nexts[i] = removedNode.nexts[i];
            }
            
            // 更新跳表的层数
            int newLevel = level;
            while (--newLevel >= 0 && first.nexts[newLevel] == null) {
                level = newLevel;
            }
            
            return removedNode.value;
        }
        
        private int randomLevel() {
            int level = 1;
            while (Math.random() < P && level < MAX_LEVEL) {
                level++;
            }
            return level;
        }
        
        private void keyCheck(K key) {
            if (key == null) {
                throw new IllegalArgumentException("key must not be null.");
            }
        }
        
        private int compare(K k1, K k2) {
            return comparator != null 
                    ? comparator.compare(k1, k2)
                    : ((Comparable<K>)k1).compareTo(k2);
        }
        
        private static class Node<K, V> {
            K key;
            V value;
            Node<K, V>[] nexts;
    //        Node<K, V> right;
    //        Node<K, V> down;
    //        Node<K, V> top;
    //        Node<K, V> left;
            public Node(K key, V value, int level) {
                this.key = key;
                this.value = value;
                nexts = new Node[level];
            }
            @Override
            public String toString() {
                return key + ":" + value + "_" + nexts.length;
            }
        }
        
        @Override
        public String toString() {
            StringBuilder sb = new StringBuilder();
            sb.append("一共" + level + "层").append("
    ");
            for (int i = level - 1; i >= 0; i--) {
                Node<K, V> node = first;
                while (node.nexts[i] != null) {
                    sb.append(node.nexts[i]);
                    sb.append(" ");
                    node = node.nexts[i];
                }
                sb.append("
    ");
            }
            return sb.toString();
        }
    }

    ...

  • 相关阅读:
    Qt音视频开发11-ffmpeg常用命令
    Qt音视频开发10-ffmpeg控制播放
    Qt音视频开发9-ffmpeg录像存储
    Intellij IDEA 与maven 版本不符 Unable to import maven project See logs for details: No implementation for org.apache.maven.model.path.PathTranslator was bound
    Java Map 遍历史上最全
    java new date 结果与操作系统时间相差8小时处理解决方法
    elasticsearch报错
    Pytorch-基于Transformer的情感分类
    Pytorch-LSTM+Attention文本分类
    PHP FFI调用go,居然比go还快
  • 原文地址:https://www.cnblogs.com/guoyu1/p/14709486.html
Copyright © 2011-2022 走看看