成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

認(rèn)識與實(shí)現(xiàn)Skip List

Yangyang / 1471人閱讀

摘要:跳表全稱叫做跳躍表,簡稱跳表。跳表在原有的有序鏈表上面增加了多級索引,通過索引來實(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è)指向下面一層的元素。

3. 來一個(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

4. 筆者的Java實(shí)現(xiàn),慢慢看,很清晰

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

相關(guān)文章

  • 轉(zhuǎn) | Java8初體驗(yàn)(二)Stream語法詳解

    摘要:第一個(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的例子可...

    taoszu 評論0 收藏0
  • 程序員的算法趣題Q39: 反復(fù)排序

    摘要:中的統(tǒng)一的終點(diǎn)是全白狀態(tài)。比如說,的第個(gè)數(shù)恰好是,它可以由根據(jù)題設(shè)規(guī)則重排而得。上一篇填充白色填充白色下一篇優(yōu)雅的地址本系列總目錄參見程序員的算法趣題詳細(xì)分析和全解程序員的算法趣題詳細(xì)分析和全解 目錄 1. 問題描述 2. 解題分析 2.1??Naive Approach--正向全量搜索 ...

    gitmilk 評論0 收藏0
  • Java多線程進(jìn)階(二五)—— J.U.C之collections框架:ConcurrentSkip

    摘要:同時(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...

    huashiou 評論0 收藏0
  • JavaScript DOM2和DOM3——“遍歷”的注意要點(diǎn)

    摘要:級遍歷和范圍模塊定義了兩個(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í)行深度...

    antz 評論0 收藏0

發(fā)表評論

0條評論

最新活動(dòng)
閱讀需要支付1元查看
<