[您有新的未分配科技點]可,可,可持久化!?------可持久化線段樹普及版講解

来源:http://www.cnblogs.com/LadyLex/archive/2017/08/02/7275164.html
-Advertisement-
Play Games

最近跑來打數據結構,於是我決定搞一發可持久化,然後發現……一發不可收啊…… 對於可持久化數據結構,其最大的特征是“歷史版本查詢”,即可以回到某一次修改之前的狀態,並繼續操作;而這種“歷史版本查詢”會衍生出其他一些強大的操作。 今天,我們主要講解可持久化線段樹。其實,它的另外一個名字“主席樹”似乎更加 ...


最近跑來打數據結構,於是我決定搞一發可持久化,然後發現……一發不可收啊……

對於可持久化數據結構,其最大的特征是“歷史版本查詢”,即可以回到某一次修改之前的狀態,並繼續操作;而這種“歷史版本查詢”會衍生出其他一些強大的操作。

今天,我們主要講解可持久化線段樹。其實,它的另外一個名字“主席樹”似乎更加為人所知(主席%%%)。

主席樹與普通的線段樹相比,多出來的操作是在修改時複製修改的一條鏈,這個操作的過程大概長下麵這樣。

至於為什麼要這樣做……

對數據進行可持久化,一種朴素的想法是每一次修改新建一棵線段樹,但是,時間複雜度和空間複雜度都是不允許我們做這樣的操作的。

我們思考一下,每一次修改,只有一條鏈上的節點被修改,而其他的節點信息都沒有變。因此,我們對這一次修改新建包括一個新根在內的logn個節點,其他的節點我們與上一課樹共用。這樣一來,我們既能保存之前的信息,又能進行修改操作。

主席樹插入操作代碼見下,有指針和數組2種版本:

指針版本:

 1 void insert(node *&a,node *b,int l,int r,int pos)
 2 {
 3     a->cnt=b->cnt+1;//cnt表示這一棵樹里的數據個數,a是新樹,b是舊樹,下同
 4     int mi=(l+r)>>1;
 5     if(l==r)return;
 6     if(pos<=mi)
 7     {
 8         a->ch[1]=b->ch[1],a->ch[0]=newnode();
 9         insert(a->ch[0],b->ch[0],l,mi,pos);
10     }
11     else 
12     {
13         a->ch[0]=b->ch[0],a->ch[1]=newnode();
14         insert(a->ch[1],b->ch[1],mi+1,r,pos);
15     }
16     a->update();
17 }

數組版本:

1 void insert(int rt1,int rt2,int l,int r,int pos)
2 {
3     if(l==r){cnt[rt1]=cnt[rt2]+1;return;}//cnt表示數據個數,rt1是新樹,rt2是舊樹
4     lc[rt1]=lc[rt2],rc[rt1]=rc[rt2];
5     int mi=(l+r)>>1;
6     if(pos<=mi)lc[rt1]=++tot,insert(lc[rt1],lc[rt2],l,mi,pos);
7     else rc[rt1]=++tot,insert(rc[rt1],rc[rt2],mi+1,r,pos);
8     cnt[rt1]=cnt[lc[rt1]]+cnt[rc[rt1]];
9 }

 

這一部分的基礎題大概有cogs2554(http://cogs.pro/cogs/problem/problem.php?pid=2554 ),這就是一道主席樹的入門水題,完全沒有瞭解的同學可以通過這道題來入門。

可能有同學註意到了上面插入代碼的cnt變數,它其實有大用:接下來,我們考慮用主席樹進行一些更高級的操作:維護靜態的區間第k大(小)值。

為什麼主席樹可以維護這個?

在對於一個序列查詢k大(小)值時,我們把主席樹建成權值線段樹,每插入一個數就新建一個版本。這時,不難看出主席樹具有區間可減性:

對於某一個區間【L,R】插入b個數以後區間中樹的個數,減去【L,R】插入a-1個數以後區間中數的個數,

結果就是區間【a,b】中權值在【L,R】的數的個數。這樣利用cnt變數不斷查詢,我們就可以找到某一個區間中第k大(小)的值

我們在代碼實現時,只需要同時傳入2個節點,並且比較區間數據個數之差與k的大小關係,進行移動即可

查詢操作代碼見下:

 1 inline int query(int l,int r,int u,int v,int k)
 2 {
 3     node *a=root[u-1],*b=root[v];
 4     while(l<r)
 5     {
 6         int tmp=b->ch[0]->cnt-a->ch[0]->cnt,mi=(l+r)>>1;
 7         if(tmp>=k)a=a->ch[0],b=b->ch[0],r=mi;
 8         else a=a->ch[1],b=b->ch[1],k-=tmp,l=mi+1;
 9     }
10     return r;
11 }

但是要註意,主席樹在不做額外處理時只能查詢靜態的區間k大(小)值。

接下來,我們就考慮動態區間k小值。如果我們要對區間進行修改的話,一個簡單的主席樹已經無法實現了。

如果對原來的節點直接修改的話,會造成不可名狀的運行錯誤(有興趣的同學可以結合上面插入代碼想一想為什麼),

空間和時間也無法接受(我們需要把後面所有樹都更改一下),但我們在做樹套樹的時候,可以做類似的操作,那麼主席樹是不是應該也套些什麼呢?

主席樹上的點,儲存的都是在一段權值區間內的數據個數,我們必須要維護數據個數才可以通過相減得到一段區間的權值線段樹。

而現在有了修改,對於這個修改的維護,朴素的做法有2種:O(1)查詢,O(n)維護(掃一遍),和O(n)查詢(現場算)和O(1)維護。

這兩種做法都不是很憂,所以我們考慮利用快捷維護首碼和的樹狀數組解決這個問題,即所謂“樹狀數組套主席樹”

如圖,圖中c節點代表對應區間的線段樹。比如,第8個點代表的線段樹就是區間[1,8]的線段樹,而第6個點代表的就是區間[5,6]的線段樹。其他點同理。

在修改時,我們用樹狀數組找出需要修改的最多logn個節點,存起來,通過同時進行的一遍修改即可解決了。查詢是類似的。

普通的樹套樹題都可以用樹狀數組套主席樹來打,這裡給出幾道練習題:

bzoj3196 二逼平衡樹 http://www.lydsy.com/JudgeOnline/problem.php?id=3196

bzoj1901 Zju2112 Dynamic Rankings(許可權題)http://www.lydsy.com/JudgeOnline/problem.php?id=1901

cogs 257 動態排名系統 http://cogs.pro/cogs/problem/problem.php?pid=257

我cogs257 的代碼 http://www.cnblogs.com/LadyLex/p/7275540.html

上面這些還沒完。更強大的是,主席樹不僅可以進行數列的維護,還可以對樹上的數據進行操作和維護。

一般來說,我們有兩種方法實現主席樹上樹:

一種是在父親節點的基礎上,在兒子節點新建樹。這樣可以維護出從某個點到根節點之間的數組,從而與LCA衍生出求樹上某一條鏈的k值問題(或其他問題)

一種是按照兩個把樹轉化為序列的工具:dfs序和歐拉序來建樹,從而解決問題。這樣可以截出一棵子樹的值,從而衍生出其他問題。

為什麼第二種方法是正確的?如果我們按照左區間+1,右區間-1的話,如果某條路徑的另一端不在鏈上中,我們就不會統計到他的答案(+1的時間過於靠前或者靠後,導致沒有統計)這個東西的確很巧妙…… 這樣一來,如果查詢點,我們把待查詢的鏈拆成[x.lca]和[y.fa[lca]]然後查詢。 如果我們統計的是邊,所以我們應該統計[x.lca]和[y.lca]。lca處的答案被統計了2次,最後記得減去; 點的查詢大概長下麵這樣

如果帶修改的話,就無法用上面第一種方法了,只能用dfs序/歐拉序轉成序列,再用樹狀數組套主席樹。插入在進棧點+1,出棧點-1。刪除在進棧點-1,出棧點+1。

一些例題:

BZOJ3123[Sdoi2013]森林 http://www.lydsy.com/JudgeOnline/problem.php?id=3123

本題我的題解:http://www.cnblogs.com/LadyLex/p/7275793.html

BZOJ3551 [ONTAK2010]Peaks加強版 http://www.lydsy.com/JudgeOnline/problem.php?id=3551

原題沒有題面(和bzoj3545是一樣的,不過bzoj3545是許可權題),可以看一下我題解里的題面(強行安利一波)http://www.cnblogs.com/LadyLex/p/7275821.html

(另外其實上面這道題的主要知識點不是主席樹,是克魯斯卡爾重構樹233)

BZOJ3772 精神污染(許可權題) http://www.lydsy.com/JudgeOnline/problem.php?id=3772

本題我的題解:http://www.cnblogs.com/LadyLex/p/7279150.html

 

除了上面這些基本操作,主席樹還可以處理一堆五花八門的問題,由於這是普及向講解以及我懶癌發作233不再一一列舉。

主席樹是一種功能強大的數據結構,通過新建鏈以及共用子節點的巧妙操作,不僅可以記錄歷史版本值,還可以支持很多其他操作。希望讀完這篇博文的你可以有所收穫!:)

另:接下來幾天我還會補充可持久化平衡樹(無旋Treap),可持久化Trie樹以及可持久化並查集的講解,敬請期待啦~


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Java的MVC模式簡介 MVC(Model View Control)模型-視圖-控制器 首先我們需要知道MVC模式並不是javaweb項目中獨有的,MVC是一種軟體工程中的一種軟體架構模式,把軟體系統分為三個基本部分:模型(Model)、視圖(View)和控制器(Controller),即為MV ...
  • 搶購、秒殺是平常很常見的場景,面試的時候面試官也經常會問到,比如問你淘寶中的搶購秒殺是怎麼實現的等等。 搶購、秒殺實現很簡單,但是有些問題需要解決,主要針對兩個問題: 1 高併發對資料庫產生的壓力 2 競爭狀態下如何解決庫存的正確減少("超賣"問題) 第一個問題,對於PHP來說很簡單,用緩存技術就可 ...
  • 我們現在介面的線上問題主要有三個,第一:啟動時有些機器會有短暫的線程池滿。第二:併發量上不去,怕服務被打死,不敢調高限流閾值。第三:499超時現象。 今天已上線 今天終於把那天說的全量執行時間延長,從圖中可以看到,中午12點發版之後,記憶體使用率有明顯下降,晚上是介面調用高峰,會有上浮,但是總體來看還 ...
  • 1、下載solr6.6 並解壓 地址: http://www.apache.org/dyn/closer.lua/lucene/solr/6.6.0 2、安裝JDK1.8 地址: http://www.oracle.com/technetwork/java/javase/downloads/jdk8 ...
  • 1 //選擇排序對數據進行升序排序 2 public static void selectSortArray(int[] arr){ 3 for(int i = 0; iarr[j]){ 6 int temp = arr[j]; 7 arr[j... ...
  • 裝箱和拆箱 裝箱和拆箱 基本數據類型的包裝類 舉兩個例子,看一下 基本數據類型的包裝類 舉兩個例子,看一下 對於byte/short/long/float/double和Integer(int)類用法類似 對於byte/short/long/float/double和Integer(int)類用法類 ...
  • 有時候,需要禁止函數修改列表。例如要對裂變進行修改操作,也要保留原來的未列印的設計列表,以供備案。為解決這個問題,可向函數傳遞列表的副本而不是原件;這樣函數所做的任何修改都隻影響副本,而絲毫不影響原件。 8-9 魔術師 魔術師 :創建一個包含魔術師名字的列表,並將其傳遞給一個名為show_magic ...
  • 垃圾收集器與記憶體分配策略 由於JVM中對象的頻繁操作是在堆中,所以主要回收的是堆記憶體,方法區中的回收也有,但是比較謹慎 一、對象死亡判斷方法 1.引用計數法 就是如果對象被引用一次,就給計數器+1,否則-1 實現簡單,但是無法解決對象相互引用的問題;實際上JVM也不是使用的此種方式,因此已下的程式我 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...