摘要:跳表全稱叫做跳躍表,簡稱跳表。跳表在原有的有序鏈表上面增加了多級索引,通過索引來實(shí)現(xiàn)快速查找。跳表不僅能提高搜索性能,同時(shí)也可以提高插入和刪除操作的性能。每個(gè)節(jié)點(diǎn)包含兩個(gè)指針,一個(gè)指向同一鏈表中的下一個(gè)元素,一個(gè)指向下面一層的元素。
前言
增加了向前指針的鏈表叫作跳表。跳表全稱叫做跳躍表,簡稱跳表。跳表是一個(gè)隨機(jī)化的數(shù)據(jù)結(jié)構(gòu),實(shí)質(zhì)就是一種可以進(jìn)行二分查找的有序鏈表。跳表在原有的有序鏈表上面增加了多級索引,通過索引來實(shí)現(xiàn)快速查找。跳表不僅能提高搜索性能,同時(shí)也可以提高插入和刪除操作的性能。
1. 跳表的樣兒 2. 跳表具有如下性質(zhì):1.由很多層結(jié)構(gòu)組成
2.每一層都是一個(gè)有序的鏈表
3.最底層(第一層)的鏈表包含所有元素
4.如果一個(gè)元素出現(xiàn)在 第 i 層 的鏈表中,則它在第 i 層 之下的鏈表也都會出現(xiàn)。
5.每個(gè)節(jié)點(diǎn)包含兩個(gè)指針,一個(gè)指向同一鏈表中的下一個(gè)元素,一個(gè)指向下面一層的元素。
栗子:查找元素 6
1.頭指針最開始在第一層最小值INT_MIN處
2.于是和旁邊的1比較,當(dāng)然了1大嘛,于是指針移向1
3.再和當(dāng)前層4比較, 比 4 大,跳到4
4.與7比較,比7小,于是向下跳到第二層4
5..與5比較,比5大,跳到5
6.與7比較,比7小,跳到最低層5
7.只能依次向后比較了,于是找到了6
1.節(jié)點(diǎn)定義,共三個(gè)成員變量:value,right,down
package crabapple; /* * Copyright (c) This is zhaoxubin"s Java program. * Copyright belongs to the crabapple organization. * The crabapple organization has all rights to this program. * No individual or organization can refer to or reproduce this program without permission. * If you need to reprint or quote, please post it to [email protected]. * You will get a reply within a week, * */ import java.util.Random; /** * 節(jié)點(diǎn)類定義 */ class Node { //值 public int value = 0; //當(dāng)前層下一個(gè)節(jié)點(diǎn) public Node right; //下一層,直連的節(jié)點(diǎn) public Node down; //構(gòu)造函數(shù) public Node() { } public Node(int value) { this.value = value; } }
2.跳表類 定義,大家可以清晰的看請代碼邏輯結(jié)構(gòu),該類只暴露insert()和search()兩個(gè)方法,其它變量及方法均設(shè)為私有.
/** * 跳表定義 */ public class SkipTable { //表層數(shù) private int levelCount; //表的頭指針 private Node firstNode; //初始化:層數(shù)為1,共前后兩個(gè)節(jié)點(diǎn),一個(gè)最小值,一個(gè)最大值 private void init() { levelCount = 1; firstNode=new Node(); firstNode.value = Integer.MIN_VALUE; firstNode.right = new Node(Integer.MAX_VALUE); } public SkipTable() { init(); } /** * 查找值 * @param value * @return */ public boolean search(int value) { Node current = firstNode; return toSearch(current, value); } private boolean toSearch(Node node, int value) { if (node.value == value) return true; else if (node.right!=null&&value >= node.right.value) return toSearch(node.right, value); else if (node.down != null) return toSearch(node.down, value); return false; } /** * 插入值 * @param value * @return */ public boolean insert(int value) { //判斷是否有這個(gè)元素 if (search(value)) return false; //隨機(jī)獲取一個(gè)層數(shù) int willLevel = updateLevelCount(); //判斷是否添加新層 if (willLevel > levelCount) { Node newFirstNode = new Node(); addLevel(firstNode, newFirstNode); firstNode=newFirstNode; levelCount = willLevel; } //插入新元素 Node port = firstNode; int skipLevel = levelCount - willLevel; //迭代到指定層 while ((skipLevel--) > 0) port = port.down; //上下層新節(jié)點(diǎn)的橋梁 Node insertNode = null; while (port != null) { //獲取當(dāng)前層第一個(gè)節(jié)點(diǎn)指針 Node curPort = port; //迭代到右邊的節(jié)點(diǎn)值比自己大為止 while (port.right.value < value) port = port.right; //準(zhǔn)備插入的新節(jié)點(diǎn) Node curInNode = new Node(value); //更新當(dāng)前節(jié)點(diǎn)和前節(jié)點(diǎn)指針指向 curInNode.right = port.right; port.right = curInNode; //將當(dāng)前節(jié)點(diǎn)引用給上層節(jié)點(diǎn) if (insertNode != null) insertNode.down = curInNode; //將新插入的節(jié)點(diǎn)指針更新到insertNode,以備在下一層建立指向 insertNode = curInNode; //行頭指針向下迭代 port = port.down; } return true; } /** * 添加新層 * * @param oldFirst * @param newFirst */ private void addLevel(Node oldFirst, Node newFirst) { newFirst.value = oldFirst.value; newFirst.down = oldFirst; if (oldFirst.right != null) { Node newRightNode = new Node(); newFirst.right = newRightNode; addLevel(oldFirst.right, newRightNode); } } /** * 以一定概率使得獲取一個(gè)和老層數(shù)差值不超過1的新層數(shù) * @return */ private int updateLevelCount() { Random random=new Random(); int v = random.nextInt(10); return v ==0 ? random.nextInt(levelCount) + 2 : random.nextInt(levelCount)+1 ; } }
3.測試類
package crabapple; public class Main { public static void main(String[] args) { //測試 SkipTable skipTable=new SkipTable(); skipTable.insert(1); skipTable.insert(2); skipTable.insert(3); skipTable.insert(4); skipTable.insert(5); skipTable.insert(6); skipTable.insert(7); skipTable.insert(8); skipTable.insert(9); skipTable.insert(10); //健壯性邊界值測試 System.out.println(skipTable.search(0)); System.out.println(skipTable.search(1)); System.out.println(skipTable.search(2)); System.out.println(skipTable.search(5)); System.out.println(skipTable.search(9)); System.out.println(skipTable.search(10)); System.out.println(skipTable.search(11)); } } //output: /** * false * true * true * true * true * true * false * * Process finished with exit code 0 */結(jié)語
上面的代碼,大家是可以直接運(yùn)行,筆者能力有限,代碼也許還有不足,望大家多多指教.
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/73978.html
摘要:第一個(gè)函數(shù)生成一個(gè)新的實(shí)例第二個(gè)函數(shù)接受兩個(gè)參數(shù),第一個(gè)是前面生成的對象,二個(gè)是中包含的元素,函數(shù)體就是把中的元素加入對象中。 感謝同事【天錦】的投稿。投稿請聯(lián)系 [email protected] 上篇文章[Java8初體驗(yàn)(一)lambda表達(dá)式語法]()比較詳細(xì)的介紹了lambda表達(dá)式的方方面面,細(xì)心的讀者會發(fā)現(xiàn)那篇文章的例子中有很多Stream的例子。這些Stream的例子可...
摘要:中的統(tǒng)一的終點(diǎn)是全白狀態(tài)。比如說,的第個(gè)數(shù)恰好是,它可以由根據(jù)題設(shè)規(guī)則重排而得。上一篇填充白色填充白色下一篇優(yōu)雅的地址本系列總目錄參見程序員的算法趣題詳細(xì)分析和全解程序員的算法趣題詳細(xì)分析和全解 目錄 1. 問題描述 2. 解題分析 2.1??Naive Approach--正向全量搜索 ...
摘要:同時(shí),也提供了一個(gè)基于的實(shí)現(xiàn)類,底層基于紅黑樹設(shè)計(jì),是一種有序的??梢钥闯墒遣l(fā)版本的,但是和不同是,并不是基于紅黑樹實(shí)現(xiàn)的,其底層是一種類似跳表的結(jié)構(gòu)。上述所有構(gòu)造器都調(diào)用了方法方法將一些字段置初始化,然后將指針指向新創(chuàng)建的結(jié)點(diǎn)。 showImg(https://segmentfault.com/img/remote/1460000016201159); 本文首發(fā)于一世流云專欄:ht...
摘要:級遍歷和范圍模塊定義了兩個(gè)用于輔助完成順序遍歷結(jié)構(gòu)的類型和這兩個(gè)類型能夠基于給定的起點(diǎn)對結(jié)構(gòu)執(zhí)行深度優(yōu)先的遍歷操作。其中的屬性,表示任何遍歷方法在上一次遍歷中返回的接待你。通過設(shè)置這個(gè)屬性也可以修改遍歷繼續(xù)進(jìn)行的節(jié)點(diǎn)。 DOM2級遍歷和范圍模塊定義了兩個(gè)用于輔助完成順序遍歷DOM結(jié)構(gòu)的類型:NodeIterator和TreeWalker;這兩個(gè)類型能夠基于給定的起點(diǎn)對DOM結(jié)構(gòu)執(zhí)行深度...
閱讀 1789·2021-11-25 09:43
閱讀 15430·2021-09-22 15:11
閱讀 2637·2019-08-30 13:19
閱讀 2019·2019-08-30 12:54
閱讀 1822·2019-08-29 13:06
閱讀 933·2019-08-26 14:07
閱讀 1622·2019-08-26 10:47
閱讀 3043·2019-08-26 10:41