bzoj1112: [POI2008]磚塊Klo(主席樹)
題目傳送門
。
解法:
比賽的時候不知道怎麼就一直在想主席樹(可能最近做主席樹有點多)
然後什麼線段樹啊樹狀陣列都沒想過。
然後就被師兄強行40M卡空間全爆了。。
大視野還是有良心的啊162M。。
就用主席樹求出中位數,然後沒了。
程式碼實現:
#include<cstdio>
#include<iostream>
#include<algorithm>
using namespace std;
typedef long long ll;
struct node {int c,lc,rc;ll s;}t[2000010];int cnt,rt[110000];
void build(int &u,int l,int r,int p,int s) {
if(u==0)u=++cnt;t[u].c++;t[u].s+=(ll)s;
if(l==r)return ;int mid=(l+r)/2;
if(p<=mid)build(t[u].lc,l,mid,p,s);
else build(t[u].rc,mid+1,r,p,s);
}
void Merge(int &u1,int u2) {
if(u1==0) {u1=u2;return ;}if(u2==0)return ;
t[u1].c+=t[u2].c;t[u1].s+=t[u2].s;
Merge(t[u1].lc,t[u2].lc);Merge(t[u1].rc,t[u2].rc);
}
int find_mid(int &u1,int &u2,int l,int r,int k) {
if(l==r)return l;int mid=(l+r)/2;
int c=t[t[u1].lc].c-t[t[u2].lc].c;
if(k<=c)return find_mid(t[u1].lc,t[u2].lc,l,mid,k);
else return find_mid(t[u1].rc,t[u2].rc,mid+1,r,k-c);
}
int find_lc(int &u1,int &u2,int l,int r,int p) {
if(t[u1].c-t[u2].c==0)return 0;
if(l==r) return t[u1].c-t[u2].c;int mid=(l+r)/2;
if(p<=mid)return find_lc(t[u1].lc,t[u2].lc,l,mid,p);
else return find_lc(t[u1].rc,t[u2].rc,mid+1,r,p)+t[t[u1].lc].c-t[t[u2].lc].c;
}
ll find_ls(int &u1,int &u2,int l,int r,int p) {
if(t[u1].s-t[u2].s==0) return 0;
if(l==r)return t[u1].s-t[u2].s;int mid=(l+r)/2;
if(p<=mid)return find_ls(t[u1].lc,t[u2].lc,l,mid,p);
else return find_ls(t[u1].rc,t[u2].rc,mid+1,r,p)+t[t[u1].lc].s-t[t[u2].lc].s;
}
struct edge {int x,id;}a[110000];ll s[110000];
bool cmp(edge n1,edge n2){return n1.x<n2.x;}
bool cmp1(edge n1,edge n2){return n1.id<n2.id;}
int main() {
//freopen("akc.in","r",stdin);freopen("akc.out","w",stdout);
int n,k;scanf("%d%d",&n,&k);int mmax=0;
for(int i=1;i<=n;i++) {scanf("%d",&a[i].x);a[i].id=i;}
if(k==1){printf("0\n");return 0;}
sort(a+1,a+1+n,cmp);int tot=0;a[0].x=-1;int tt=0;
for(int i=1;i<=n;i++) if(a[i].x!=a[i-1].x)tt++;
for(int i=1;i<=n;i++) {if(a[i].x!=a[i-1].x)tot++;build(rt[a[i].id],1,tt,tot,a[i].x);s[tot]=a[i].x;}
for(int i=1;i<=n;i++) Merge(rt[i],rt[i-1]);
sort(a+1,a+1+n,cmp1);ll T=0;for(int i=1;i<k;i++)T+=ll(a[i].x);ll Ans=ll(100000000000);a[0].x=0;
for(int i=k;i<=n;i++) {
T+=ll(a[i].x);T-=ll(a[i-k].x);
int p=find_mid(rt[i],rt[i-k],1,tt,(k+1)/2);
int lc=find_lc(rt[i],rt[i-k],1,tt,p),rc=k-lc;
ll ls=find_ls(rt[i],rt[i-k],1,tt,p),rs=T-ls;
Ans=min(Ans,ll(lc*s[p])-ls+rs-ll(rc*s[p]));
}printf("%lld\n",Ans);
return 0;
}
相關文章
- 主席樹
- 主席樹模板
- 靜態主席樹模板
- 動態主席樹模板
- HDU2665 Kth number【主席樹】
- HDU4417 Super Mario【主席樹】
- BZOJ4299: Codechef FRBSUM(主席樹)
- D-query SPOJ - DQUERY (主席樹)
- 新一代打磚塊遊戲《星際磚塊》開發分享遊戲
- 【資料結構】淺談主席樹資料結構
- 洛谷P4197 Peaks(Kruskal重構樹 主席樹)
- bzoj3524: [Poi2014]Couriers(主席樹)
- HDU-4348 - To the moon (主席樹+區間修改)
- 【主席數】可持續化線段樹
- SDOI2018 原題識別(主席樹)
- 演算法隨筆——主席樹(可持久化線段樹)演算法持久化
- 【主席樹】P3919 【模板】可持久化線段樹 1持久化
- bzoj3545: [ONTAK2010]Peaks(主席樹+最小生成樹)
- bzoj4477: [Jsoi2015]字串樹(主席樹+Hash+Lca)JS字串
- bzoj3439: Kpm的MC密碼(主席樹+DFS序+字典樹)密碼
- bzoj2809: [Apio2012]dispatching(DFS序+主席樹)API
- 求區間不同數的個數【主席樹求解】
- bzoj5178: [Jsoi2011]棒棒糖(主席樹)JS
- bzoj2588: Spoj 10628. Count on a tree(主席樹+LCA)
- 2024年3月21日 懸繩法 + 珂朵莉樹(ODT) + 主席樹
- [題解] [洛谷 P1174] 打磚塊
- 可持久化線段————主席樹(洛谷p3834)持久化
- bzoj3110: [Zjoi2013]K大數查詢(主席樹+樹狀陣列)陣列
- CFTC主席:對區塊鏈的監管“落伍”了區塊鏈
- 洛谷 P3919 可持久化線段樹 1 之主席樹模板(初級)持久化
- bzoj1146: [CTSC2008]網路管理Network(dfs序+主席樹+樹狀陣列)陣列
- bzoj1803: Spoj1487 Query on a tree III(DFS序+主席樹)
- bzoj4448: [Scoi2015]情報傳遞(主席樹+Lca)
- bzoj5177: [Jsoi2013]貪心的導遊(主席樹)JS
- bzoj3207: 花神的嘲諷計劃Ⅰ(hash+主席樹)
- [POI2008] POC-Trains 題解AI
- bzoj2733: [HNOI2012]永無鄉(並查集+主席樹)並查集
- bzoj3932: [CQOI2015]任務查詢系統(主席樹)