K Seq HihoCoder - 1046 || BZOJ4504 k個串

来源:https://www.cnblogs.com/hehe54321/archive/2018/03/18/8598141.html
-Advertisement-
Play Games

這題與超級鋼琴類似,然而重覆的不重覆計算貢獻。。 那麼先求出數組nxt,nxt[i]表示第i個元素之後的第一個與其相等的元素的下標,不存在則nxt[i]=0 考慮取的區間左端點為1時的情況。 將讀入序列a中相等元素都只保留最先出現的,其餘變為0,然後求首碼和,得到數組b。 此時可以知道,設f(l,r ...


這題與超級鋼琴類似,然而重覆的不重覆計算貢獻。。

那麼先求出數組nxt,nxt[i]表示第i個元素之後的第一個與其相等的元素的下標,不存在則nxt[i]=0

考慮取的區間左端點為1時的情況。

將讀入序列a中相等元素都只保留最先出現的,其餘變為0,然後求首碼和,得到數組b。

此時可以知道,設f(l,r)為取下標在[l,r]區間內數時的答案,那麼f(1,r)=b[r]。

考慮取的區間左端點為2時的情況。如何維護b數組,使得新的b數組也滿足f(2,r)=b[r]?

手模樣例區間左端點為1和2時符合要求的b。

樣例:

7 5
3 -2 1 2 2 1 3

l=1:  3  1  2 4 4 4 4
l=2:  0  -2 -1 1 1 1 4

可以發現,做的操作相當於將b數組內下標在[1,nxt[1]-1]區間內的數減了(原來的)b[1]。

進一步推出,左端點為l時的b數組變到左端點l+1的b數組,就相當於將下標在[l,nxt[l]-1]區間內的數減了(原來的)b[l]。

(當然如果nxt[i]=0,那麼就是將[l,n]內減b[l])

因此,可以用可持久化線段樹處理出左端點為每一個位置時的b數組。可持久化線段樹如果要傳標記的話常數會很大(複雜度應該是對的...大概吧?),所以可以標記永久化

(不知道為什麼網上的標記永久化那麼長?嚇得我還以為自己寫錯了233333)

(我寫標記永久化時候困難重重,最後發現標記含義的定義還是類比普通線段樹最容易實現(就是除當前節點外,以當前節點為根的子樹內所有點的對應值都應加上當前節點的加法tag),另外給標記和記錄的值一個明確的定義對於寫清楚代碼非常重要)

這樣子之後就可以用類似超級鋼琴的做法去做了...才怪。難道你真的跟我一樣想要去寫可持久化的帶區間修改的區間第k大這樣子一看就不靠譜的東西?

可以發現每一次對給定某一個b數組的查詢,每次的區間都是一樣的,一定是第一次第1大,第二次第2大,...

當然就可以用超級鋼琴的後一種做法去做了(優先隊列維護一下哪個b數組,哪一段區間的最大值是多少,在哪裡,當某一段區間最大值被取出後,把該區間除了最大值所在位置剩下的最多兩段取最大值放回優先隊列)。'

(似乎也可以考慮暴力將原來的最大值加一個-inf?這樣子下一次查找找到的就是該區間的次大啦(我沒試過))

錯誤記錄:

1.不知道為什麼想到去維護最小值了,怎麼過的樣例啊

2.82-83行i+1寫成i

  1 #include<cstdio>
  2 #include<algorithm>
  3 #include<queue>
  4 #include<tr1/unordered_map>
  5 #define inf 0x3f3f3f3f3f3f3f3f
  6 #define mid ((l+r)>>1)
  7 using namespace std;
  8 using namespace tr1;
  9 typedef long long LL;
 10 typedef pair<LL,LL> P;
 11 LL lc[30000010],rc[30000100],addv[30000100],mem;
 12 //addv[i]表示i區間加法tag
 13 P maxn[30000100];//一對值:區間(只考慮自身及其下節點的標記)的最大值及下標
 14 LL L,R,x;
 15 LL root[100100],nxt[100100],a[100100],b[100100];
 16 tr1::unordered_map<LL,LL> ttt1;
 17 void build(LL l,LL r,LL &num)
 18 {
 19     num=++mem;
 20     if(l==r)    {maxn[num]=P(b[l],l);return;}
 21     build(l,mid,lc[num]);build(mid+1,r,rc[num]);
 22     maxn[num]=max(maxn[lc[num]],maxn[rc[num]]);
 23 }
 24 void addx(LL l,LL r,LL &num)
 25 {
 26     LL t=num;num=++mem;lc[num]=lc[t];rc[num]=rc[t];maxn[num]=maxn[t];addv[num]=addv[t];
 27     if(L<=l&&r<=R)
 28     {
 29         addv[num]+=x;maxn[num].first+=x;
 30         return;
 31     }
 32     if(L<=mid)    addx(l,mid,lc[num]);
 33     if(mid<R)    addx(mid+1,r,rc[num]);
 34     maxn[num]=max(maxn[lc[num]],maxn[rc[num]]);
 35     maxn[num].first+=addv[num];
 36 }
 37 P query(LL l,LL r,LL num)
 38 {
 39     if(L<=l&&r<=R)    return maxn[num];
 40     P ans=P(-inf,1);
 41     if(L<=mid)    ans=max(ans,query(l,mid,lc[num]));
 42     if(mid<R)    ans=max(ans,query(mid+1,r,rc[num]));
 43     ans.first+=addv[num];
 44     return ans;
 45 }
 46 struct Info
 47 {
 48     Info(LL a=0,LL b=0,LL c=0,LL d=0,LL e=0)
 49         :ans(a),l(b),r(c),st(d),pos(e)
 50     {}
 51     LL ans,l,r,st,pos;
 52     //root[st]中,[l,r]內最大值是ans,在pos位置
 53     friend bool operator<(const Info &a,const Info &b)
 54     {
 55         return a.ans<b.ans;
 56     }
 57 };
 58 LL n,k;
 59 priority_queue<Info> q;
 60 int main()
 61 {
 62     LL i;P t;Info t2;
 63     scanf("%lld%lld",&n,&k);
 64     for(i=1;i<=n;i++)    scanf("%lld",&a[i]);
 65     for(i=1;i<=n;i++)
 66     {
 67         b[i]=b[i-1];
 68         if(ttt1.count(a[i])==0)
 69             b[i]+=a[i];
 70         else
 71             nxt[ttt1[a[i]]]=i;
 72         ttt1[a[i]]=i;
 73     }
 74     build(1,n,root[0]);
 75     L=1,R=n,t=query(1,n,root[0]);
 76     q.push(Info(t.first,1,n,0,t.second));
 77     for(i=1;i<n;i++)
 78     {
 79         root[i]=root[i-1];
 80         L=i,R=i,x=-query(1,n,root[i]).first;
 81         L=i,R=nxt[i]?nxt[i]-1:n,addx(1,n,root[i]);
 82         L=i+1/**/,R=n,t=query(1,n,root[i]);
 83         q.push(Info(t.first,i+1,n,i,t.second));//
 84     }
 85     for(i=1;i<k;i++)
 86     {
 87         t2=q.top();q.pop();
 88         L=t2.l,R=t2.pos-1;
 89         if(L<=R)
 90         {
 91             t=query(1,n,root[t2.st]);
 92             q.push(Info(t.first,L,R,t2.st,t.second));
 93         }
 94         L=t2.pos+1,R=t2.r;
 95         if(L<=R)
 96         {
 97             t=query(1,n,root[t2.st]);
 98             q.push(Info(t.first,L,R,t2.st,t.second));
 99         }
100     }
101     t2=q.top();
102     printf("%lld",t2.ans);
103     return 0;
104 }

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

-Advertisement-
Play Games
更多相關文章
  • 2018-03-19 子類可以繼承父類的對象方法。在繼承後,重覆提供該方法,就叫做方法的重寫;又叫覆蓋 override 1、父類Item 父類Item有一個方法,叫做effect 2、子類LifePotion 子類LifePotion繼承Item,同時也提供了方法effect 3、調用重寫的方法 ...
  • IntelliJ Idea 常用快捷鍵列表 Ctrl+Shift + Enter,語句完成“!”,否定完成,輸入表達式時按 “!”鍵Ctrl+E,最近的文件Ctrl+Shift+E,最近更改的文件Shift+Click,可以關閉文件Ctrl+[ OR ],可以跑到大括弧的開頭與結尾Ctrl+F12, ...
  • 2018-03-19 在設計LOL的時候,進攻類英雄有兩種,一種是進行物理系攻擊,一種是進行魔法系攻擊。這時候,就可以使用介面來實現這個效果。 介面就像是一種約定,我們約定某些英雄是物理系英雄,那麼他們就一定能夠進行物理。 一、物理攻擊介面 創建一個介面 File->New->Interface A ...
  • 索引表設計 在電商項目中,物理庫存系統是個極其重要的系統,訂單支付後,就會開始來占用物理庫存。一般情況下,庫存系統都是要分庫的,因為主要的操作是寫操作,例如占用/釋放/取消等寫操作。使用分庫可以降低資料庫寫的壓力。儘管寫操作為主,但是讀操作也是有的。比如說,庫存占用的時候,得先查詢是否有庫存,而這個 ...
  • 最近到廣州某建站互聯網公司面試,當時面試官問假設有兩個字元串String a="abc",String b = "abc";問輸出a==b是true還是false。我當時毫不猶豫答了true,然後根據字元串常量池的知識點結合jvm的記憶體模型講解,然而他卻跟我說是false,說這是最基本的問題。我當時 ...
  • 使用的 Python 版本 3.6.4 0.>>>import this(挺有趣的命令,覺得其他語言也應該加一些類似的文檔,時刻提醒) 1.Python把非空字元串解讀為True(if SOMETHING NOTEMPTY:) 2.切片複製 clone_list = list_name[:](防止修 ...
  • 準備 : jdk包和MyEclipse安裝包 1.點擊jdk安裝包 , 然後一直點下一步 , 點到最後就OK了 2.安裝MyEclipse : 1>雙擊MyEclipse 然後點next 2>接受按裝協議 先選中 I accept the terms of the license agreement ...
  • 確認本機已安裝python環境 查看pip版本 安裝psutil 卸載第三方庫 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...