在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

C語言跳轉表的實現

C語言編程 ? 來源:博客園 ? 作者:pdudos ? 2022-10-19 14:23 ? 次閱讀

跳表比較好理解,但是實際用代碼來表示,還是有點復雜的。

實現的方法不唯一

1、什么是跳表

跳表是鏈表+索引的一種數據結構 ,是以空間換取時間的方式。

2、跳表概念

跳表在原有鏈表的基礎上,增加索引,從而可以進行二分查找,提高搜尋效率。

原始鏈表

Head ——> 1 ——> 8 ——> 12 ——> 23 ——> 55 ——> NULL

新增了索引的鏈表(跳表)

Head2 ————————> 8 ———————————————————————> NULL 
Head1 ————————> 8 —————————> 23 —————————> NULL 
Head0 ——> 1 ——> 8 ——> 12 ——> 23 ——> 55 ——> NULL

Head0 , Head1 , Head2 上都是真實的節點,這就是以空間換取時間

例如算上Head, 元素數據一共有 6 個,而添加索引后,元素一共有 11 個

3、跳表增刪查規則

3.1 跳表數據節點

數據節點可以和鏈表節點一致 ,也可以定義如下節點,除了數據外,有指針指向 前一個/后一個/上一個/下一個 節點,以便后續查找操作。

typedef struct {
        int data;
        struct Node *next; // 后一個節點
        struct Node *last; // 前一個節點
        struct Node *up; // 上一個節點
        struct Node *down; // 下一個節點
} Node;

3.2 跳表初始化

當跳表有多少層的時候,應當建立多少個頭結點,例如: 跳表為3層

Head2 ——> NULL
Head1 ——> NULL
Head0 ——> NULL

3.3 查找

刪除/新增 都會進行查詢才操作,無非是刪除/新增索引而已。

例如有如下數據

Head2 —————————————————————> 23 —————————> NULL 
Head1 ————————> 8 —————————> 23 —————————> NULL 
Head0 ——> 1 ——> 8 ——> 12 ——> 23 ——> 55 ——> NULL

要查找13這個節點

去除無效層

例如:Head2后面第一個節點的數據23, 而23大于13, 所以Head2沒有數據匹配查詢,故需要跳到下面一層,至Head1上進行查詢。

查詢至Head0層

去除無效層后數據進入了Head1, 在Head1上進行匹配,當匹配到23時,23大于13,將23標記為 查詢結束點,對23的上一個節點 8 進行 向下指針操作,進入Head0層的8節點。

查找實際數據

Head0層的8 進行查找,直至查詢結束標記點(head1 23), 查詢的數據分別為 8 , 12 ,23 查詢結束,未找到數據。

3.4 新增

新增操作需要記錄索引尋址過程,以便后續新增索引。

頭結點插入

頭結點插入一定是去除無效層至Head0 , 且Head0的第一個節點都比插入節點要大的情況下

例如:

如下跳表,插入 2

Head2 —————————————————————> 23 —————————> NULL 
Head1 ————————> 8 —————————> 23 —————————> NULL 
Head0 ——> 3 ——> 8 ——> 12 ——> 23 ——> 55 ——> NULL

尾結點插入

頭結點插入一定是去除無效層至Head0 , 且Head0的第一個節點都比插入節點要小,直至NULL節點的情況下

例如:

如下跳表,插入 65

Head2 —————————————————————> 23 —————————> NULL 
Head1 ————————> 8 —————————> 23 —————————> NULL 
Head0 ——> 3 ——> 8 ——> 12 ——> 23 ——> 55 ——> NULL

中間節點插入

除開以上2種情況,其余情況為 中間節點插入

新增索引

拋硬幣的方法,當數據量達到一定規模的時候,一定是趨近于 50%的。

所以跳表會越來越趨向于如下形式

    3
    3       7
1   3   5   7   9
1 2 3 4 5 6 7 8 9

判斷是否需要新增索引,采取拋硬幣的方法來判斷,即: 隨機數 取余 為 0 則需要新增,否則不需要。

例如如下跳表,插入 65

Head2 —————————————————————> 23 —————————> NULL 
Head1 ————————> 8 —————————> 23 —————————> NULL 
Head0 ——> 3 ——> 8 ——> 12 ——> 23 ——> 55 ——> NULL

尋址應該為
Head2: 23
Head1: 23

元素數據插入后為

Head2 —————————————————————> 23 ———————————————> NULL 
Head1 ————————> 8 —————————> 23 ———————————————> NULL 
Head0 ——> 3 ——> 8 ——> 12 ——> 23 ——> 55 ——> 65 —> NULL  

當插入65節點后,若判斷需要索引的時候,則先為 Head1 添加索引,添加位置為 尋址地址之后,寄 Head1: 23

Head2 —————————————————————> 23 ———————————————> NULL 
Head1 ————————> 8 —————————> 23 —————————> 65 —> NULL 
Head0 ——> 3 ——> 8 ——> 12 ——> 23 ——> 55 ——> 65 —> NULL  

繼續判斷,若不需要添加索引,則插入結束

若還需要添加索引,則繼續上述操作,直至 索引層 達到最高層

3.5 刪除

刪除首先是查找操作【3.3 查找】

若未找到該節點,則刪除失敗

若找到了該節點,則應當提到該數據最高索引層,再從高到低刪除

例如:

如下跳表,刪除 23

Head2 —————————————————————> 23 ———————————————> NULL 
Head1 ————————> 8 —————————> 23 —————————> 65 —> NULL 
Head0 ——> 3 ——> 8 ——> 12 ——> 23 ——> 55 ——> 65 —> NULL  

找到 Head0 23 后,應該向上找到 Head2 23 ,然后從高向低刪除,若刪除后,該索引沒有數據了,則索引層減1

則刪除Head2 23 后數據如下

Head1 ————————> 8 —————————> 23 —————————> 65 —> NULL 
Head0 ——> 3 ——> 8 ——> 12 ——> 23 ——> 55 ——> 65 —> NULL  

刪除Head1 23 后數據如下

Head1 ————————> 8 ———————————————————————> 65 —> NULL 
Head0 ——> 3 ——> 8 ——> 12 ——> 23 ——> 55 ——> 65 —> NULL 

刪除Head0 23后數據如下

Head1 ————————> 8 ————————————————> 65 —> NULL 
Head0 ——> 3 ——> 8 ——> 12 ——> 55 ——> 65 —> NULL 

4、代碼

skipList.c

# include 
# include 
# include 

int MaxLevel = 8; // 最大層數
int currLevel = 0; // 當前層數

// 數據節點
typedef struct {
        int data;
        struct Node *next;
        struct Node *last;
        struct Node *up;
        struct Node *down;
} Node;

// 記錄索引尋址過程
typedef struct {
        int level;
        struct Node *node;
} skipStep;

// 判斷是否需要新增索引, 拋硬幣
bool randNum() {
        if(0 == (rand() % 2))
                return true;

        return false;
}

// 新增節點
bool add(Node *SL[] , int data) {

        printf("新增節點: %d
",data);
        int level = currLevel;

        Node *Head = NULL;
        Node *tmp = NULL;
        Node *last = NULL;

        // 初始化索引 數據為 Head 地址
        skipStep steps[MaxLevel];
        int i;
        for (i=0;inext;

        while ((level > 0) && (data < tmp->data)) {
                level--;
                Head = SL[level];
                tmp = Head->next;
        }

        // 根據索引尋找Head0數據節點
        while ((level > 0)) {

                while (tmp != NULL) {
                        if (data < tmp->data) {
                                steps[level].level = level;
                                if (NULL != last) 
                                        steps[level].node = last;
                                tmp = last->down;
                                level--;
                                break;
                        }

                        last = tmp;
                        tmp = tmp->next;
                }
                if (NULL == tmp) {
                        steps[level].level = level;
                        if (NULL != last) 
                                steps[level].node = last;
                        tmp = last->down;
                        level--;

                }
        }

        // Head0 數據合適的節點
        while (tmp != NULL) {
                if (data < tmp->data) {
                        break;
                }
                last = tmp;
                tmp = tmp->next;
        }

        // 新增節點
        Node *newData = (Node *)malloc(sizeof(Node));
        newData->data = data;
        newData->up = NULL;
        newData->down = NULL;
        newData->last = NULL;
        newData->next = NULL;

        int k = 0;

        // Head0 插入原始數據
        if (NULL == last ) {
                // 頭結點

                Head = SL[0];
                Node *headNext = Head->next;
                if (NULL != headNext) {
                        newData->next = headNext;
                        headNext->last = newData;

                        newData->last = Head;


                } 

                Head->next = newData;
                newData->last = Head;


        } else if ( NULL == tmp) {
                // 尾節點
                last->next = newData;
                newData->last = last;


        } else {
                // 中間節點
                newData->next = tmp;
                tmp->last = newData;

                newData->last = last;
                last->next = newData;
        }

        // 構建索引
        while (randNum()) {
                k++;
                if (k >= MaxLevel) break;

                // 新增索引數據
                Node *newIndex = (Node *)malloc(sizeof(Node));
                newIndex->data = data;
                newIndex->up = NULL;
                newIndex->down = NULL;
                newIndex->next = NULL;
                newIndex->last = NULL;

                // 建立上下級關系
                newIndex->down = newData;
                newData->up = newIndex;

                Node *node = steps[k].node;

                // node->next
                Node *nextIndex = node->next;


                node->next = newIndex;
                newIndex->last = node;

                newIndex->next = nextIndex;
                if (NULL != nextIndex) 
                        nextIndex->last = newIndex;

                newData = newIndex;

                // 判斷是否需要新增索引層數
                if (k > currLevel) 
                        currLevel = k;
        }
}


// 初始化頭結點
Node *initSkipList(Node *skipList[]) {
        int i;
        for (i=0;idata = -1-i;
                newHead->down = NULL;
                newHead->up = NULL;
                newHead->next = NULL;
                newHead->last = NULL;

                skipList[i] = newHead;


        }
        return skipList;
}

// 打印跳表數據
void PrintSkipList(Node *SL[]) {
        if (NULL == SL) {
                return;
        };

        int level = currLevel;
        //int level = MaxLevel;

        int i;
        for (i=level;i>=0;i--) {
                Node *Head = SL[i];

                Node *tmp = Head->next;
                printf("第%d層		",i);
                while (NULL != tmp) {
                        printf(" %d	",tmp->data);

                        tmp = tmp->next;
                }
                printf("
");
        }
}

// 查詢數據
Node *query(Node *SL[] , int data) {
        printf("查詢數據: %d
",data);

        int level = currLevel;

        Node *Head = NULL;
        Node *tmp = NULL;
        Node *last = NULL;

        Head = SL[level];
        tmp = Head->next;
        int endQuery = -1;

        // 篩除無效層
        while ((level > 0) && (data < tmp->data)) {
                level--;
                endQuery = tmp->data;
                Head = SL[level];
                tmp = Head->next;
        }

        // 根據索引定位到Head0層
        while ((level > 0 )) {

                while (tmp != NULL) {
                        if (data < (tmp->data)) {
                                level--;
                                endQuery = tmp->data;
                                tmp = last->down;
                                break;
                        }

                        last = tmp;
                        tmp = tmp->next;
                }
                if (NULL == tmp) {
                        tmp = last->down;
                        endQuery = -1;
                        level--;
                }

        }

        // 查詢實際數據
        while (NULL != tmp) {
                if (endQuery != -1)
                        if (tmp->data > endQuery) {
                                        tmp = NULL;
                                        break;
                        }
                if (tmp->data == data) {
                        break;
                }
                tmp = tmp->next;
        }
        // 返回查詢的數據節點,若沒有查詢到,應當返回NULL ,否則返回實際的地址
        return tmp;
}

// 刪除數據
bool del(Node *SL[],int data) {
        printf("刪除數據: %d
",data);

        // 找到節點地址
        Node *tmp = query(SL,data);

        if (NULL == tmp) {
                printf("未找到節點,刪除失敗
");
                return false;
        }
        int level = 0;
        Node *t_last = NULL;
        Node *t_next = NULL;


        // 找到該數據最高索引
        while (NULL != tmp->up) {
                level++;
                tmp = tmp->up;
        }

        // 由上至下刪除索引/數據
        while (tmp != NULL) {
                t_last = tmp->last;
                t_next = tmp->next;

                Node *t_down = tmp->down;

                if (t_last == NULL) {
                        printf("上一個節點不可能為空,刪除失敗,層數: %d
",level);
                        return false;
                }

                t_last->next = t_next;

                if (NULL != t_next)
                        t_next->last = t_last;
                else
                        t_last->next = NULL;

                if ((t_last == SL[level]) && (NULL == t_next)) {
                        currLevel--;

                }
                free(tmp);

                tmp = t_down;
                level--;
        }

        return true;


}

int main() {

        Node *SL[MaxLevel];

        Node *skipList = initSkipList(SL);
        if (NULL == SL) {
                printf("skipList 申請失敗
");
                return -1;
        }

        // 測試新增
        int num[] = {1,3,2,10,8,9,22,30,29,120,99,78,55,76,21};
        int i;
        for (i=0;i

執行結果

# gcc skipList.c -w -g
# ./a.out 
新增節點: 1
新增節點: 3
新增節點: 2
新增節點: 10
新增節點: 8
新增節點: 9
新增節點: 22
新增節點: 30
新增節點: 29
新增節點: 120
新增節點: 99
新增節點: 78
新增節點: 55
新增節點: 76
新增節點: 21
第5層            99
第4層            99
第3層            76      99
第2層            9       76      99
第1層            3       9       29      30      76      78      99
第0層            1       2       3       8       9       10      21      22      29      30      55      76      78      99      120
刪除數據: 99
查詢數據: 99
刪除數據: 9
查詢數據: 9
刪除數據: 78
查詢數據: 78
刪除數據: 55
查詢數據: 55
刪除數據: 3
查詢數據: 3
刪除數據: 1
查詢數據: 1
刪除數據: 28
查詢數據: 28
未找到節點,刪除失敗
刪除數據: 78
查詢數據: 78
未找到節點,刪除失敗
第3層            76
第2層            76
第1層            29      30      76
第0層            2       8       10      21      22      29      30      76      120

#
審核編輯:湯梓紅
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 代碼
    +關注

    關注

    30

    文章

    4798

    瀏覽量

    68728
  • 數據結構
    +關注

    關注

    3

    文章

    573

    瀏覽量

    40152

原文標題:代碼小技巧-跳轉表

文章出處:【微信號:C語言編程,微信公眾號:C語言編程】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    嵌入式C語言中的goto語句詳解

    goto語句被稱為C語言中的跳轉語句。用于無條件跳轉到其他標簽。它將控制權轉移到程序的其他部分。
    發表于 07-19 16:08 ?3403次閱讀
    嵌入式<b class='flag-5'>C</b><b class='flag-5'>語言</b>中的goto語句詳解

    求助關于匯編語言的兩道題~

    1.用匯編語言編寫含有2個分支的跳轉實現程序跳轉。R2寄存器中存放的是跳轉
    發表于 04-23 23:20

    匯編語言 跳轉 問題

    我是51單片機的初學者,匯編語言不熟悉。現在有這么一個小問題R0這種寄存器的跳轉命令?可以用什么?我上網查過了JCC命令,不是有錯誤就是不識別謝謝
    發表于 10-12 19:08

    C語言:標準和實現

    C 語言:標準與實現 The Standards and Implementations of the C Programming Language前言
    發表于 11-27 22:27 ?58次下載

    IA-64二進制翻譯中跳轉恢復技術

    基于IA-64體系結構下二進制翻譯系統,本文提出了應用過程內靜態切片技術恢復索引跳轉跳轉以及目標地址的解決方案。并通過在IA-64體系結構上對C
    發表于 08-29 10:16 ?14次下載

    CRC算法原理及C語言實現

    CRC算法原理及C語言實現:本文從理論上推導出CRC 算法實現原理,給出三種分別適應不同計算機或微控制器硬件環境的C 語言程序。讀者更能根據
    發表于 09-23 23:38 ?31次下載

    pid算法原理和C語言的簡單實現

    pid算法原理和C語言的簡單實現,有興趣的可以看看
    發表于 12-07 18:34 ?9次下載

    C語言教程之打印乘法口訣問題

    C語言教程之打印乘法口訣問題,很好的C語言資料,快來學習吧。
    發表于 04-25 15:03 ?0次下載

    C語言教程之不使用strcpy()函數實現

    C語言教程之不使用strcpy()函數實現,很好的C語言資料,快來學習吧。
    發表于 04-25 15:03 ?0次下載

    C語言標準與實現

    C語言標準與實現 HENBUCUO BJBUI
    發表于 06-08 18:18 ?0次下載

    C語言模擬實現strcat函數

    C語言模擬實現strcat函數
    的頭像 發表于 06-29 16:18 ?2447次閱讀

    C語言模擬實現strcmp函數

    C語言模擬實現strcmp函數
    的頭像 發表于 06-29 16:51 ?2687次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>模擬<b class='flag-5'>實現</b>strcmp函數

    累加校驗和C語言實現

    累加校驗和C語言實現
    發表于 11-29 18:06 ?10次下載
    累加校驗和<b class='flag-5'>C</b><b class='flag-5'>語言實現</b>

    C 語言跳轉實現及在嵌入式設備中的應用

    In computer programming, a branch table or jump table is a method of transferring program control (branching) to another part o...
    發表于 02-07 11:34 ?0次下載
    <b class='flag-5'>C</b> <b class='flag-5'>語言</b><b class='flag-5'>跳轉</b><b class='flag-5'>表</b>的<b class='flag-5'>實現</b>及在嵌入式設備中的應用

    怎么用C語言實現多態

    這里我想主要介紹下在C語言中是如何實現的面向對象。知道了C語言實現面向對象的方式,我們再聯想下,C
    的頭像 發表于 10-12 09:12 ?2067次閱讀
    主站蜘蛛池模板: 国产亚洲精品久久久久久牛牛| 美女大黄三级视频在线观看| 国产精品夜夜春夜夜爽| 九色福利| 黄色大片免费观看| 国产精品国产三级国产在线观看| 色女人网| 中文天堂在线视频| 亚洲综合色站| 天天碰人人| lsj老司机精品视频在线观看| japanese色系tube日本护士| 亚洲国产七七久久桃花| 久久精品免费看| 2021国产精品| 成人免费久久精品国产片久久影院| 国产激情久久久久影院小草| 欧美成人三级网站| ww在线观看| 亚洲天堂bt| 一级毛片免费网站| 日本三级免费| 国语自产自拍秒拍在线视频| 又黑又长黑人欧美三级| 欧美黑粗硬| 1515hh四虎免费观38com| 在线精品国产三级| 青草悠悠视频在线观看| 国产精品15p| 日本欧美强乱视频在线| 天天爱天天做天天爽| 国产情侣露脸| 日本在线观看一区| 婷婷久久综合九色综合九七| 午夜剧场黄| 一级aa 毛片高清免费看| 日韩一级欧美一级在线观看| 色噜噜狠狠狠色综合久| 六月丁香六月婷婷| 午夜影院404| 末发育娇小性色xxxxx视频|