4504: K个串

Time Limit: 20 Sec  Memory Limit: 512 MB
Submit: 268  Solved: 110
[Submit][Status][Discuss]

Description

兔子们在玩k个串的游戏。首先,它们拿出了一个长度为n的数字序列,选出其中的一
个连续子串,然后统计其子串中所有数字之和(注意这里重复出现的数字只被统计一次)。
兔子们想知道,在这个数字序列所有连续的子串中,按照以上方式统计其所有数字之和,第
k大的和是多少。

Input

第一行,两个整数n和k,分别表示长度为n的数字序列和想要统计的第k大的和
接下里一行n个数a_i,表示这个数字序列

Output

一行一个整数,表示第k大的和

Sample Input

7 5
3 -2 1 2 2 1 3 -2

Sample Output

4

HINT

1 <= n <= 100000, 1 <= k <= 200000, 0 <= |a_i| <= 10^9数据保证存在第 k 大的和

看到这道题感觉一脸不可做,直到有人告我主席树是可以区间修改的,靠我以前一直以为不行的。。。

思路还是很简单的,和超级钢琴一样,用堆存一个五元组,表示一个右端点往左某个区间的局部最大值,每次分裂成两个,暴力找出前k大的就好了。

所以要用主席树维护一个区间min和支持区间打标记,这道题的标记可以永久化。

据yousiki说也可以支持push_down,只是查询的时候需要新建节点。

因为第一次写所以出现了很多神奇的错误。

最开始修改我是这么写的:

if(ll<=l&&rr>=r)
{
a[x].id=a[y].id;
a[x].lazy=a[y].lazy+z;
a[x].mn=a[x].lazy;
return ;
}

果断过不了样例,查着查着意识到好像区间修改和单点修改不太一样,x应该有左右儿子的!于是:

if(ll<=l&&rr>=r)
{
  a[x].id=a[y].id;
  a[x].lazy=a[y].lazy+z;
  a[x].mn=a[x].lazy;
  a[x].l=a[y].l;
  a[x].r=a[y].r;
  return ;
}

过样例后交上去wa,拍拍拍,拍出来后发现不对,x又不是叶子,它的min和lazy怎么可能相等!!!

if(lll<=l&&rr>=r)
{
a[x].id=a[y].id;
a[x].lazy=a[y].lazy+z;
a[x].mn=a[x].lazy+a[y].mn;
a[x].l=a[y].l;
a[x].r=a[y].r
return ;
}

小数据都拍过了,很高兴试了组大数据,不给面子的wa了,直觉告诉我还是这错了,我靠我在写什么,这顺序不对啊,我给x的mn加了两倍的y.lazy.

if(lll<=l&&rr>=r)
{
a[x].id=a[y].id;
a[x].lazy=a[y].lazy+z;
a[x].mn=z+a[y].mn;
a[x].l=a[y].l;
a[x].r=a[y].r
return ;
}

普天同庆,终于对了。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
#include<vector>
#define N 100005
#define ll long long
#define inf 1LL<<61
using namespace std;
int n,k;
struct node
{
int l,r,pos,rr;
ll zhi;
node(){;}
node(int _l,int _r,int _pos,int _rr,ll _zhi)
{
l=_l;r=_r;pos=_pos;rr=_rr;zhi=_zhi;
}
friend bool operator < (const node &aa,const node &bb)
{
return aa.zhi<bb.zhi;
}
};
priority_queue<node>q;
struct tree
{
int l,r,id;
ll lazy,mn;
tree()
{
lazy=mn=0;
}
}a[N*30]; int cnt;
int root[N];
int b[N];
vector<int>v[N];
void push_up(int x)
{
if(a[a[x].l].mn<a[a[x].r].mn)
{
a[x].mn=a[a[x].l].mn;
a[x].id=a[a[x].l].id;
}
else
{
a[x].mn=a[a[x].r].mn;
a[x].id=a[a[x].r].id;
}
a[x].mn+=a[x].lazy;
}
void build(int x,int l,int r)
{
a[x].id=l;
if(l==r)return ;
int mid=(l+r)>>1;
a[x].l=++cnt;a[x].r=++cnt;
build(a[x].l,l,mid);build(a[x].r,mid+1,r);
}
void insert(int x,int y,int l,int r,int lll,int rr,int z)
{
if(lll<=l&&rr>=r)
{
a[x].id=a[y].id;
a[x].lazy=a[y].lazy+z;
a[x].mn=z+a[y].mn;
a[x].l=a[y].l;
a[x].r=a[y].r;
return ;
}
a[x].lazy=a[y].lazy;
int mid=(l+r)>>1;
if(lll<=mid)
{
a[x].l=++cnt;
insert(a[x].l,a[y].l,l,mid,lll,rr,z);
}
else a[x].l=a[y].l;
if(rr>mid)
{
a[x].r=++cnt;
insert(a[x].r,a[y].r,mid+1,r,lll,rr,z);
}
else a[x].r=a[y].r;
push_up(x);
return ;
}
int ans1;ll ans2;
void qur(int x,int l,int r,int lll,int rr,ll now)
{
if(lll<=l&&rr>=r)
{
ll tmp=now+a[x].mn;
if(tmp<ans2)
{
ans1=a[x].id;ans2=tmp;
}
return ;
}
int mid=(l+r)>>1;
if(lll<=mid)qur(a[x].l,l,mid,lll,rr,now+a[x].lazy);
if(rr>mid)qur(a[x].r,mid+1,r,lll,rr,now+a[x].lazy);
return ;
}
ll sum[N];
int li[N],ct,nxt[N],now[N];
int main()
{
scanf("%d%d",&n,&k);
for(int i=1;i<=n;i++)scanf("%d",&b[i]),li[++ct]=b[i];
sort(li+1,li+ct+1);
ct=unique(li+1,li+ct+1)-li-1;
for(int i=1;i<=n;i++)b[i]=lower_bound(li+1,li+ct+1,b[i])-li;
for(int i=1;i<=ct;i++)now[i]=n+1;
for(int i=n;i>=1;i--)
{
nxt[i]=now[b[i]];
v[nxt[i]].push_back(i);
now[b[i]]=i;
}
memset(now,0,sizeof(now));
for(int i=1;i<=n;i++)
{
sum[i]=sum[i-1];
if(!now[b[i]])
{
now[b[i]]=1;
sum[i]+=li[b[i]];
}
}
cnt=0;
root[n+2]=++cnt;
build(1,0,n-1);
for(int i=n+1;i>=1;i--)
{
root[i]=root[i+1];
for(int j=0;j<v[i].size();j++)
{
if(v[i][j]==n)continue;
int tmp=root[i];root[i]=++cnt;
insert(root[i],tmp,0,n-1,v[i][j],n-1,li[b[v[i][j]]]);
}
}
for(int i=1;i<=n;i++)
{
ans2=inf;qur(root[i+1],0,n-1,0,i-1,0);
q.push(node(0,i-1,ans1,i,sum[i]-ans2));
}
for(int i=1;i<=k;i++)
{
node tmp=q.top();q.pop();
if(tmp.l!=tmp.pos)
{
ans2=inf;qur(root[tmp.rr+1],0,n-1,tmp.l,tmp.pos-1,0);
q.push(node(tmp.l,tmp.pos-1,ans1,tmp.rr,sum[tmp.rr]-ans2));
}
if(tmp.r!=tmp.pos)
{
ans2=inf;qur(root[tmp.rr+1],0,n-1,tmp.pos+1,tmp.r,0);
q.push(node(tmp.pos+1,tmp.r,ans1,tmp.rr,sum[tmp.rr]-ans2));
}
if(i==k)
{
printf("%lld\n",tmp.zhi);
}
}
return 0;
}

  

  

bzoj : 4504: K个串 区间修改主席树的更多相关文章

  1. bzoj 4504: K个串 可持久化线段树+堆

    题目: Description 兔子们在玩k个串的游戏.首先,它们拿出了一个长度为n的数字序列,选出其中的一 个连续子串,然后统计其子串中所有数字之和(注意这里重复出现的数字只被统计一次). 兔子们想 ...

  2. BZOJ 1146: [CTSC2008]网络管理Network 带修改主席树_树套树_DFS序

    Description M公司是一个非常庞大的跨国公司,在许多国家都设有它的下属分支机构或部门.为了让分布在世界各地的N个 部门之间协同工作,公司搭建了一个连接整个公司的通信网络.该网络的结构由N个路 ...

  3. BZOJ 1901: Zju2112 Dynamic Rankings | 带修改主席树

    题目: emmmm是个权限题 题解: 带修改主席树的板子题,核心思想是用树状数组维护动态前缀和的性质来支持修改 修改的时候修改类似树状数组一样进行logn个Insert 查询的时候同理,树状数组的方法 ...

  4. bzoj 4504: K个串【大根堆+主席树】

    像超级钢琴一样把五元组放进大根堆,每次取一个出来拆开,(d,l,r,p,v)表示右端点为d,左端点区间为(l,r),最大区间和值为v左端点在p上 关于怎么快速求区间和,用可持久化线段树维护(主席树?) ...

  5. BZOJ 4504: K个串

    题目大意: 求一个序列的第k大的子串和. 题解: 对于一个右端点找最优的左端点,扔进堆里. 每次取堆顶,将这个右端点可以选择的左端点的区间分成两段,扔进堆里,重复k次. 现在需要对于一个固定的右端点, ...

  6. 少年,想学带修改主席树吗 | BZOJ1901 带修改区间第k小

    少年,想学带修改主席树吗 | BZOJ1901 带修改区间第k小 有一道题(BZOJ 1901)是这样的:n个数,m个询问,询问有两种:修改某个数/询问区间第k小. 不带修改的区间第k小用主席树很好写 ...

  7. [bzoj P4504] K个串

    [bzoj P4504] K个串 [题目描述] 兔子们在玩k个串的游戏.首先,它们拿出了一个长度为n的数字序列,选出其中的一个连续子串,然后统计其子串中所有数字之和(注意这里重复出现的数字只被统计一次 ...

  8. 【bzoj1901】dynamic ranking(带修改主席树/树套树)

    题面地址(权限题) 不用权限题的地址 首先说说怎么搞带修改主席树? 回忆一般的kth问题,我们的主席树求的是前缀和,这样我们在目标区间的左右端点的主席树差分下就能求出kth. 那么我们如何支持修改操作 ...

  9. [BZOJ3295] [Cqoi2011]动态逆序对(带修改主席树)

    题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任务是在每次删除一个元素之前统计整个序列的逆序 ...

随机推荐

  1. Tomcat源码学习(3)—— Digester介绍

    Digester方法详解: 通读Digester之前先分析下他的结构: 1.1该类继承了方法DefaultHandler2,DefaultHandler2继承了DefaultHandler是和sax解 ...

  2. Scrum Meeting 11.11

    成员 今日任务 明日计划 用时 徐越 学习UI设计并重构上传下载界面     赵庶宏       薄霖 UI代码更新     卞忠昊 Xfermode与PorterDuff     武鑫 界面设计:独 ...

  3. 第二阶段Sprint冲刺会议9

    进展:查看有关“共享平台”的资料,看如何实现上传下载功能,并尝试编码,没有成功.

  4. Task 4 求数组的连续子数组的最大和(团队合作)

    小组成员:李敏.刘子晗 1.设计思想:由于已经做过这个题目,只要对之前的程序加上相应的测试和约束即可.我们两个人一起商议后,决定了程序的主框架和并列出了最终可以实现的功能.先要定义数组长度和上下限的变 ...

  5. matconvnet编译

    1.安装matconvnet 网上教程很多 2.编译 cd matconvnet/ addpath matlab/ vl_compilenn('enableGpu',true,'cudaRoot',' ...

  6. 关于mybatis的思考(1)——mybatis的使用实例

    架构分析 MyBatis 是支持普通 SQL 查询,存储过程和高级映射的优秀持久层框架. MyBatis 消除了几乎所有的 JDBC 代码和参数的手工设置以及对结果集的检索.MyBatis 可以使用简 ...

  7. week4a:个人博客作业

    本周结对项目的要求: 黄金点游戏是一个数字小游戏,其游戏规则是: N个同学(N通常大于10),每人写一个0~100之间的有理数 (不包括0或100),交给裁判,裁判算出所有数字的平均值,然后乘以0.6 ...

  8. EasyUi模糊匹配搜索框combobox

    现在项目当中很多已经应用了Jquery-easyUi这个界面框架了,所以,学习一点easyUI的常用工具就显得很重要了,现在介绍的就是我在项目中用到的easyUi的模糊匹配组合框combobox. c ...

  9. SDWebImage缓存图片的机制

    SDWebImage是一个很厉害的图片缓存的框架.既ASIHttp+AsyncImage之后,我一直使用AFNetworking集成的UIImageView+AFNetworking.h,但后者对于图 ...

  10. 关于Keil C关键字xdata和data的问题

    1.xdata表示这是一个外部RAM地址内的数据,数据最终将被保存至外部RAM的某个地址单元中:但是,外部RAM只能通过寄存器间接寻址来访问,也就是说,其地址需要保存在内部RAM中(其实或许是SFR中 ...