返回顶部
首页 > 资讯 > 后端开发 > Python >Java数据结构之实现跳表
  • 864
分享到

Java数据结构之实现跳表

2024-04-02 19:04:59 864人浏览 薄情痞子

Python 官方文档:入门教程 => 点击学习

摘要

目录一、跳表的定义二、跳表搜索三、插入元素四、删除元素五、完整代码一、跳表的定义 跳跃表是一种随机化数据结构,基于并联的链表,其效率可比拟于二叉查找树(对于大多数操作需要O(log

一、跳表的定义

跳跃表是一种随机化数据结构,基于并联的链表,其效率可比拟于二叉查找树(对于大多数操作需要O(log n)平均时间),并且对并发算法友好。

SkipList(跳表)是一种可以代替平衡树的数据结构,默认是按照Key值升序的。SkipList让已排序的数据分布在多层链表中,以0-1随机数决定一个数据的向上攀升与否,通过“空间来换取时间”的一个算法,在每个节点中增加了向前的指针,在插入、删除、查找时可以忽略一些不可能涉及到的结点,从而提高了效率。

在Java的api中已经有了实现:分别是:

ConcurrentSkipListMap(在功能上对应HashTable、HashMap、TreeMap) ;
ConcurrentSkipListSet(在功能上对应HashSet).
确切来说,SkipList更像Java中的TreeMap,TreeMap基于红黑树(一种自平衡二叉查找树)实现的,时间复杂度平均能达到O(log n)。
HashMap是基于散列表实现的,时间复杂度平均能达到O(1)。ConcurrentSkipListMap是基于跳表实现的,时间复杂度平均能达到O(log n)。

SkipList的性质

(1) 由很多层结构组成,level是通过一定的概率随机产生的。
(2) 每一层都是一个有序的链表,默认是升序
(3) 最底层(Level 1)的链表包含所有元素。
(4) 如果一个元素出现在Level i 的链表中,则它在Level i 之下的链表也都会出现。
(5) 每个节点包含两个指针,一个指向同一链表中的下一个元素,一个指向下面一层的元素。

图示:

在这里插入图片描述

二、跳表搜索

在这里插入图片描述

例子:查找元素 117

(1) 比较 21, 比 21 大,往后面找

(2) 比较 37, 比 37大,比链表最大值小,从 37 的下面一层开始找

(3) 比较 71, 比 71 大,比链表最大值小,从 71 的下面一层开始找

(4) 比较 85, 比 85 大,从后面找

(5) 比较 117, 等于 117, 找到了节点。



    private node findPreNode(int val){
        Node first = head.getFirst();//从最上层的头节点开始搜索
        while (first!=null){
            if(first.data < val && first.next.data > val){
                if(first.down == null){break;}
                first = first.down;//往下搜索
            }else if(first.data < val && first.next.data < val){
                first = first.next;//往右搜索
            }else if(first.data < val && first.next.data == val){
                return first;
            }
        }
        return null;
    }

三、插入元素

先确定该元素要占据的层数 K(采用丢硬币的方式,这完全是随机的)然后在 Level 1 … Level K 各个层的链表都插入元素。



    private int getLeavel(){
        int k = 0;
        while(rd.nextInt(2) == 1){
            k ++;
        }
        return k;
    }

例子:插入 119, K = 2

在这里插入图片描述

如果 K 大于链表的层数,则要添加新的层。
例子:插入 119, K = 4

在这里插入图片描述

四、删除元素

从上往下删除

在这里插入图片描述



    boolean delete(int val){
        Node lindPre = findPreNode(val);//找到待删除元素的最上层的前一个节点
        if(lindPre == null){
            return false;
        }
        while (true){
            lindPre.next = lindPre.next.next;
            lindPre = lindPre.down;//往下遍历,直到最底下一层
            if(lindPre==null){break;}//跳出循环
            //找到待删除元素所在行的前一个节点
            while (lindPre.next.data != val){
                lindPre = lindPre.next;
            }
        }
        size--;
        return true;
    }

五、完整代码


package com.longstudy.alGorithm;
import java.util.LinkedList;
import java.util.Random;


public class SkipList {

    //使用头插法插入新节点
    LinkedList<Node> head;//每一行的头结点,相当于跳表的第一列, 默认设置为 Integer.MIN_VALUE
    LinkedList<Node> tail;//每一行大最后一个节点,相当与跳表的最后一列  Integer.MAX_VALUE
    Random rd ;//用于生成随机数数
    int hight=-1;//当前跳表的层数,hight从0开始,初始值为-1,
    int size;//所有的节点数

    public SkipList(){
        this.head = new LinkedList<>();
        this.tail = new LinkedList<>();
        this.rd = new Random();
    }

    public static void main(String[] args) {
        SkipList sl = new SkipList();
        int[] arr = new int[500];
        for (int i = 0; i < 500; i++) {
            arr[i] = (int)(Math.random()*600);
        }
        sl.arrayToSkipList(arr);
        sl.showSkipList();
        System.out.println(sl.find(100));
        System.out.println(sl.find(50));
        System.out.println(sl.find(99));
        System.out.println("清空跳表");
        sl.clear();
        sl.showSkipList();

    }
    
    private class Node{
        int data;//存放数据
        Node next;//指向右边节点
        Node down; //指向下面节点
        int level;//当前所在的层
        public Node(){}
        public Node(int data,int level){
            this.data = data;
            this.level = level;
        }
        public Node(int data,int level,Node next,Node down){
            this.data = data;
            this.level = level;
            this.next = next;
            this.down =down;
        }
    }

    
    boolean add(int val){
        int k = getLeavel();//获得层数
        //层数比当前大的时候,增加新的层
        if(k>hight){
            int i = k-hight;
            for (int j = 1; j <=i; j++) {
                //新增头节点和尾节点
                Node h = new Node(Integer.MIN_VALUE,hight+j);
                if(head.size()>0){
                    h.down = head.getFirst();//往下指
                }
                Node t = new Node(Integer.MAX_VALUE,hight+j);//尾
                if(tail.size()>0){
                    t.down = tail.getFirst();
                }
                h.next=t;//头指向尾
                tail.addFirst(t);
                head.addFirst(h);
            }
            hight =k;//修改当前的跳表层数
        }
        return addFromK(val,k);//从第k层添加元素
    }

    
    boolean addFromK(int val,int k){
        Node preNewNode = new Node(val,k);
        Node preLine = head.get(hight-k);//获取新增节点所在层的头节点
        while (preLine != null){
            while (preLine.next.data < val){//往右搜索
                preLine = preLine.next;
            }
            preNewNode.next = preLine.next;
            preLine.next = preNewNode;
            //如果不是第一层,则建立下一层的新节点
            if (preNewNode.level>0){
                Node newNode = new  Node(val,preNewNode.level-1);
                preNewNode.down = newNode;//往下指向新节点
                preNewNode = newNode;
            }
            //往下层建立新节点
            preLine = preLine.down;
        }
        size++;//跳表中的元素数量加一
        return true;
    }


    
    private int getLeavel(){
        int k = 0;
        while(rd.nextInt(2) == 1){
            k ++;
        }
        return k;
    }

    
    boolean delete(int val){
        Node lindPre = findPreNode(val);//找到待删除元素的最上层的前一个节点
        if(lindPre == null){
            return false;
        }
        while (true){
            lindPre.next = lindPre.next.next;
            lindPre = lindPre.down;//往下遍历,直到最底下一层
            if(lindPre==null){break;}//跳出循环
            //找到待删除元素所在行的前一个节点
            while (lindPre.next.data != val){
                lindPre = lindPre.next;
            }
        }
        size--;
        return true;
    }

    
    void clear(){
        this.hight=-1;
        this.size =0;
        this.head = null;
        this.tail = null;

    }

    
    boolean find(int val){
       return findPreNode(val) !=null;
    }

    
    private Node findPreNode(int val){
        Node first = head.getFirst();//从最上层的头节点开始搜索
        while (first!=null){
            if(first.data < val && first.next.data > val){
                if(first.down == null){break;}
                first = first.down;//往下搜索
            }else if(first.data < val && first.next.data < val){
                first = first.next;//往右搜索
            }else if(first.data < val && first.next.data == val){
                return first;
            }
        }
        return null;
    }

    
    void arrayToSkipList(int[] arr){
        int len = arr.length;
        for (int i = 0; i < len; i++) {
            add(arr[i]);
        }
    }

    
    void showSkipList(){
        System.out.println("元素个数为:"+size);
        //从上往下逐层打印
        for (int i = 0; i <=hight ; i++) {
            Node linFirst = head.get(i);
            System.out.print("第"+linFirst.level+"层:\t"+"head ->\t");
            linFirst = linFirst.next;//跳过第一列的元素
            while (linFirst != null){
                if(linFirst.next != null){
                    System.out.print(""+linFirst.data +'\t'+"->\t");//+ " height:"+linFirst.level
                }else {
                    System.out.println("tail");
                }
                linFirst = linFirst.next;
            }
            System.out.println();
        }
    }

}

到此这篇关于Java数据结构之实现跳表的文章就介绍到这了,更多相关Java跳表内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: Java数据结构之实现跳表

本文链接: https://lsjlt.com/news/126663.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Java数据结构之实现跳表
    目录一、跳表的定义二、跳表搜索三、插入元素四、删除元素五、完整代码一、跳表的定义 跳跃表是一种随机化数据结构,基于并联的链表,其效率可比拟于二叉查找树(对于大多数操作需要O(log ...
    99+
    2024-04-02
  • golang怎么实现跳表数据结构
    跳表是一种基于链表的数据结构,与平衡树类似,可以实现快速的查找、插入和删除操作。跳表是由William Pugh于1990年提出的,它的实现是基于链表,在链表的基础上增加多级索引,从而可以通过索引快速地定位到链表的节点。跳表底层的链表可以是...
    99+
    2023-05-14
  • Java数据结构之顺序表的实现
    目录前言一、顺序表1.1 什么是顺序表二、简单实现顺序表2.1 创建顺序表2.2 打印顺序表2.3 获取顺序表长度2.4 在 pos 位置新增元素2.5 判定是否包含某个元素2.6 ...
    99+
    2024-04-02
  • Java数据结构之LinkedList从链表到实现
    目录1.ArrayList的缺陷2.LinkedListLinkedList概念LinkedList的使用3.链表的概念及结构4.ArrayList和LinkedList的区别1.A...
    99+
    2023-05-18
    Java LinkedList Java 链表
  • Java数据结构之双向链表的实现
    目录1 双向链表1.1 双向链表介绍1.2 双向链表实现思路2 双向链表实现完整代码2.1 节点类 Student.java2.2 双向链表实现类 StudentDoubleLink...
    99+
    2022-11-13
    Java 数据结构 双向链表 Java 双向链表
  • Redis数据结构之跳跃表是什么
    小编给大家分享一下Redis数据结构之跳跃表是什么,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!前言跳跃表是一种有序的数据结构,...
    99+
    2024-04-02
  • Java数据结构之ArrayList从顺序表到实现
    目录1 ArrayList2 ArrayList使用2.1 ArrayList的构造2.2 ArrayList常见操作2.3 ArrayList的遍历2.4 ArrayList的扩容...
    99+
    2023-05-18
    Java ArrayList Java ArrayList顺序表
  • Java数据结构之双向链表如何实现
    这篇文章主要讲解了“Java数据结构之双向链表如何实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java数据结构之双向链表如何实现”吧!双向链表(Doubly linked list)什...
    99+
    2023-06-30
  • 用Python实现数据结构之链表
    链表与栈,队列不一样,它是由一个个节点构成的,每个节点存储着本身的一些信息,也存储着其他一个或多个节点的引用,可以从一个节点找到其他的节点,节点与节点之间就像是有链连在一起一样,这种数据结构就叫做链表 单向链表是链表的最简单形式,链表...
    99+
    2023-01-30
    数据结构 链表 Python
  • Python数据结构与算法之跳表详解
    目录0. 学习目标1. 跳表的基本概念1.1 跳表介绍1.2 跳表的性能1.3 跳表与普通链表的异同2. 跳表的实现2.1 跳表结点类2.2 跳表的初始化2.3 获取跳表长度2.4 ...
    99+
    2024-04-02
  • Java数据结构之链表的概念及结构
    目录1、链表的概念2、结点3、链表的使用场景4、链表分类和常用结构5、与顺序表的比较1、链表的概念 概念:链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链...
    99+
    2023-05-14
    Java 数据结构链表概念结构 数据结构链表概念 数据结构链表结构
  • Java数据结构之链表详解
    目录一、链表的介绍二、单链表的实现三、双向链表的实现四、循环链表的实现五,链表相关的面试题一、链表的介绍 什么是链表 链表是一种物理存储单元上非连续、非顺序的存储结构,数据元素的逻辑...
    99+
    2024-04-02
  • Java数据结构之顺序表篇
    目录一.线性表 二.顺序表1.概念及结构2.顺序表的实现打印顺序表获取顺序表的有效长度在pos位置新增元素判断是否包含某个元素查找某个元素对应的位置获取/查找pos位置的元...
    99+
    2024-04-02
  • C++数据结构之单链表的实现
    目录一、单链表的定义二、单链表的基本操作的实现1.初始化2.取值3.查找4.插入5.删除三、完整代码四、测试一下代码一、单链表的定义 线性表的链式存储又称为单链表,它是指通过一组任意...
    99+
    2024-04-02
  • 数据结构TypeScript之链表实现详解
    目录链表结构特点面向对象方法封装链表构造函数基本单元:链表节点主体:链表查找节点增加节点删除节点链表结构特点 链表是线性表的其中一种,用于存储有固定顺序的元素。而元素之间会通过&r...
    99+
    2023-01-30
    TypeScript数据结构链表 TypeScript数据结构
  • C++数据结构之哈希表的实现
    目录哈希表概念散列函数直接定址法除留余数法平方取中法哈希冲突线性探测二次探测链地址法哈希表的实现闭散列开散列哈希表概念 二叉搜索树具有对数时间的表现,但这样的表现建立在一个假设上:输...
    99+
    2023-03-11
    C++数据结构 哈希表 C++哈希表 C++数据结构
  • Java数据结构之快速幂的实现
    目录引入具体方法代码实现题目矩阵快速幂斐波那契数列第 N 个泰波那契数统计元音字母序列的数目引入 快速幂是用来解决求幂运算的高效方式。 例如我们要求 x 的&nb...
    99+
    2024-04-02
  • Java数据结构之KMP算法的实现
    目录问题介绍暴力求解知识补充Next示例Next代码匹配示例匹配代码完整代码本次我们介绍数据结构中的KMP算法,我们会从下面几个角度来介绍: 问题介绍 首先我们先介绍适用于KMP算法...
    99+
    2022-11-21
    Java KMP算法 Java KMP
  • Java数据结构之并查集的实现
    目录代码解析代码应用实际应用并查集就是将原本不在一个集合里面的内容合并到一个集合中。 在实际的场景中用处不多。 除了出现在你需要同时去几个集合里面查询,避免出现查询很多次,从而放在一...
    99+
    2024-04-02
  • 数据结构(Java实现)LinkedList与链表(上)
    链表 逻辑结构 无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等。 无头双向链表:在Java的集合框架库中LinkedList底层实现就是无头双向循环链表。 ...
    99+
    2023-08-30
    数据结构 java 链表
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作