FHQTreap:實現快速查找和插入操作的二叉搜索樹

一、什麼是FHQTreap

FHQTreap是一種基於Treap樹的二叉搜索樹,它在Treap樹的基礎上做了優化,可以實現快速的查找和插入操作。FHQTreap最大的特點是,它使用了兩種隨機性的策略來維護數列中的元素,一種是隨機優先級值,一種是隨機旋轉。FHQTreap是由FHQ在2016年發明的,相較於其他數據結構而言,FHQTreap具有更優秀的時間和空間複雜度。

下面我們來看看如何實現FHQTreap。首先需要定義FHQTreap中存儲的節點結構體:

struct Node {
    int key, val;
    int size, pri;
    Node* ch[2];
    Node(int K=0, int V=0) : key(K), val(V), size(1), pri(rand()) { ch[0] = ch[1] = NULL; }
};

其中,key是存儲的關鍵字,val是存儲的值,size代表以該節點為根節點的子樹大小,pri是節點的隨機優先級值,ch數組用來存儲該節點的左右兒子。在初始化Node結構體時,pri值為隨機生成的。

二、如何實現FHQTreap樹的查找

FHQTreap樹的查找操作和二叉搜索樹的查找操作類似,都是從根節點開始,按照二叉搜索樹的規則查找左右兒子。不同之處在於,FHQTreap樹通過比較節點的優先級值來進行查找操作。

導航函數find操作實現如下:

Node* find(Node* u, int v) {
    while (u != NULL) {
        if (u -> val == v) return u;
        u = u -> ch[u -> key < v];
    }
    return u;
}

這是一個標準的二叉搜索樹的遞歸實現,它在每次遞歸時都會比較u節點和v的大小,返回左右子節點進行遞歸查找。不同之處在於,我們可以通過比較節點的優先級值來判斷查找方向,先朝條件更優秀的方向查找。

三、如何實現FHQTreap樹的插入

插入操作是FHQTreap樹中的最重要操作,它使用了兩種隨機性策略:隨機優先級值和隨機旋轉。先看看插入函數的基本寫法:

void insert(Node* &u, Node* v) {
    if (u == NULL) u = v;
    else {
        int k = (u -> key  key);
        insert(u -> ch[k], v);
        u -> size = u -> ch[0] -> size + u -> ch[1] -> size + 1; // 更新節點的size
        if (u -> ch[k] -> pri > u -> pri) rotate(u, k); // 旋轉
    }
}

這是一個遞歸操作,在每個節點上進行比較,如果該節點為空,直接將v節點插入即可。否則,遞歸查找左右兒子,同時更新節點的sizesize值,以便於後續的操作。如果該節點的子節點的優先級大於它本身的優先級,進行旋轉操作,使得樹的平衡性得以維護。

旋轉操作是FHQTreap樹中的核心操作,它用來保證樹的平衡性。旋轉分為左旋和右旋。左旋操作如下:

void rotate(Node* &u, int k) {
    Node* v = u -> ch[k];
    u -> ch[k] = v -> ch[k ^ 1];
    v -> ch[k ^ 1] = u;
    u -> size = u -> ch[0] -> size + u -> ch[1] -> size + 1;
    v -> size = v -> ch[0] -> size + v -> ch[1] -> size + 1;
    u = v;
}

在左旋操作中,我們先找到u節點和v節點,並將u的右節點指向v的左節點,v的左節點指向u,然後重新計算節點的size值。

右旋操作與左旋操作類似,這裡不再贅述。

四、如何實現FHQTreap樹的刪除

刪除操作是FHQTreap樹中的另外一個重要操作,它用來將數列中的元素刪除,在每次刪除操作後,我們可以確保該節點已經不存在於樹中了。

由於FHQTreap樹的刪除操作需要分為兩步:將待刪除節點旋轉到葉節點,然後將其刪除,我們需要先實現旋轉到葉節點的操作。rotateUntilLeaf如下:

Node* rotateUntilLeaf(Node* u, int k) {
    while (u -> ch[k] != NULL) {
        if (u -> ch[k] -> ch[k ^ 1] != NULL) {
            int dir = (u -> ch[k] -> ch[k ^ 1] -> pri > u -> ch[k] -> pri);
            rotate(u -> ch[k], dir ^ 1);
        }
        rotate(u, k);
    }
    return u;
}

在該操作中,我們從指定的節點開始,逐層旋轉深入直到到達終止節點(即該節點的左右兒子均為空),期間我們還需要調整節點的優先級值,旋轉方向等信息,以達到旋轉的目的。

其中的刪除函數erase如下:

void erase(Node* &u, int v) {
    if (u == NULL) return;
    if (u -> val == v) {
        if (u -> ch[0] == NULL && u -> ch[1] == NULL) u = NULL;
        else if (u -> ch[0] == NULL) u = u -> ch[1];
        else if (u -> ch[1] == NULL) u = u -> ch[0];
        else {
            int k = (u -> ch[0] -> pri > u -> ch[1] -> pri);
            rotate(u, k);
            erase(u -> ch[k ^ 1], v);
        }
    }
    else {
        erase(u -> ch[u -> val  size = u -> ch[0] -> size + u -> ch[1] -> size + 1;
}

該函數通過比較節點的值來進行查找,如果找到了待刪除節點,那麼將其旋轉到樹的葉子節點處,然後進行刪除操作。最後更新節點的size信息即可。

五、FHQTreap的應用

FHQTreap是一種基於Treap樹的二叉搜索樹,它在Treap樹的基礎上做了優化,可以實現快速的查找和插入操作。FHQTreap最大的特點是,它使用了兩種隨機性的策略來維護數列中的元素,一種是隨機優先級值,一種是隨機旋轉。相較於其他數據結構而言,FHQTreap具有更優秀的時間和空間複雜度。

FHQTreap的應用非常廣泛,比如說用途一:實現單調隊列。單調隊列是一種特殊的隊列,它支持兩種操作:入隊和出隊。入隊操作是將一個元素插入到隊列尾部,而出隊操作是將隊列頭部的元素彈出。FHQTreap可以使用雙端隊列來實現單調隊列,並且其時間複雜度為O(1)。

比如說用途二:統計區間內的滿足條件的元素個數。假設我們有一個序列A,需要統計A[1…n]中滿足某個條件的元素數量,那麼可以使用FHQTreap來解決。首先對序列A建立FHQTreap,然後對序列A中的每一個元素,查找FHQTreap中比它小的元素的數量,就是它在序列A中的rank值。以此來計算區間內的滿足條件的元素數量,相比於暴力查找,時間複雜度有了明顯的提升。

六、總結

本文介紹了FHQTreap的基礎知識和操作,包括何為FHQTreap樹、如何實現FHQTreap樹的查找和插入操作、如何實現FHQTreap樹的刪除操作以及FHQTreap的常見應用。值得注意的是,FHQTreap的高效性是基於它使用了兩種隨機性的策略維護數列中的元素。學習FHQTreap需要較紮實的C++編程能力,同時還需要對Treap樹的基本知識有所掌握。

原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-hant/n/270677.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
小藍的頭像小藍
上一篇 2024-12-16 13:38
下一篇 2024-12-16 13:38

相關推薦

  • Python棧操作用法介紹

    如果你是一位Python開發工程師,那麼你必須掌握Python中的棧操作。在Python中,棧是一個容器,提供後進先出(LIFO)的原則。這篇文章將通過多個方面詳細地闡述Pytho…

    編程 2025-04-29
  • Python操作數組

    本文將從多個方面詳細介紹如何使用Python操作5個數組成的列表。 一、數組的定義 數組是一種用於存儲相同類型數據的數據結構。Python中的數組是通過列表來實現的,列表中可以存放…

    編程 2025-04-29
  • Python操作MySQL

    本文將從以下幾個方面對Python操作MySQL進行詳細闡述: 一、連接MySQL數據庫 在使用Python操作MySQL之前,我們需要先連接MySQL數據庫。在Python中,我…

    編程 2025-04-29
  • Python磁盤操作全方位解析

    本篇文章將從多個方面對Python磁盤操作進行詳細闡述,包括文件讀寫、文件夾創建、刪除、文件搜索與遍歷、文件重命名、移動、複製、文件權限修改等常用操作。 一、文件讀寫操作 文件讀寫…

    編程 2025-04-29
  • Python代碼實現迴文數最少操作次數

    本文將介紹如何使用Python解決一道經典的迴文數問題:給定一個數n,按照一定規則對它進行若干次操作,使得n成為迴文數,求最少的操作次數。 一、問題分析 首先,我們需要了解迴文數的…

    編程 2025-04-29
  • Python元祖操作用法介紹

    本文將從多個方面對Python元祖的操作進行詳細闡述。包括:元祖定義及初始化、元祖遍歷、元祖切片、元祖合併及比較、元祖解包等內容。 一、元祖定義及初始化 元祖在Python中屬於序…

    編程 2025-04-29
  • 如何用Python對數據進行離散化操作

    數據離散化是指將連續的數據轉化為離散的數據,一般是用於數據挖掘和數據分析中,可以幫助我們更好的理解數據,從而更好地進行決策和分析。Python作為一種高效的編程語言,在數據處理和分…

    編程 2025-04-29
  • Python列表的讀寫操作

    本文將針對Python列表的讀取與寫入操作進行詳細的闡述,包括列表的基本操作、列表的增刪改查、列表切片、列表排序、列表反轉、列表拼接、列表複製等操作。 一、列表的基本操作 列表是P…

    編程 2025-04-29
  • Python序列的常用操作

    Python序列是程序中的重要工具,在數據分析、機器學習、圖像處理等很多領域都有廣泛的應用。Python序列分為三種:列表(list)、元組(tuple)和字符串(string)。…

    編程 2025-04-28
  • Python獲取Flutter上內容的方法及操作

    本文將從以下幾個方面介紹Python如何獲取Flutter上的內容: 一、獲取Flutter應用數據 使用Flutter提供的Platform Channel API可以很容易地獲…

    編程 2025-04-28

發表回復

登錄後才能評論