[bzoj P4504] K个串

【题目描述】

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

兔子们想知道,在这个数字序列所有连续的子串中,按照以上方式统计其所有数字之和,第k大的和是多少。

【输入格式】

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

接下里一行n个数a_i,表示这个数字序列

【输出格式】

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

【样例输入】

7 5

3 -2 1 2 2 1 3 -2

【样例输出】

4

【数据范围】

对于20%的数据,1 <= n <= 2000

对于另外20%的数据,0 <= a_i <= 10^9

对于100%的数据,1 <= n <= 100000, 1 <= k <= 200000, 0 <= |a_i| <= 10^9

数据保证存在第k大的和

题外话:好久都没有用博客园啦,最近写博客都写在WordPress上面。没想到今天炸了。。只好先来cnblogs避一避了。

对于这一题我也是一脸懵逼式的弃疗。连想都没怎么想,丝毫没有办法。

结果——主席树+堆。也是看了题解才明白的。自己怎么也想不到这种方法。

主思路是这样的,维护一个五元组(v,l,r,x,p)表示一个状态,分别表示当前状态所对应区间的和(v),左端点所在区间(l,r),左端点的具体位置(x),右端点的具体位置(p)。(这个思路骑士真的太难想到了,主要一个我觉得,习惯于考虑对称的东西,不会像这样,区间左右端点有别)

先不考虑如何如何构造或得到五元组。假设我们可以很快得到某一个特定的五元组。那如何得出第k大的和?

每一次从堆中取出最大值,也就是当前最大的和,然后做这样的事情:

构造五元组(maxsum_val(),l,p-1,maxsum_pos(),p)和(maxsum_val(),p+1,r,maxsum_pos(),p),并将它们push入堆中。显然这是正确的。

那刚开始在堆里的是什么呢?当然是p=1~n时,左端点在某一点能使区间和最大的这个状态咯。

这个问题——涉及到区间修改,区间查询。肯定要用线段树咯。但是我们发现用线段树是无法处理对于不同的右端点p,查询某个区间最值的问题的。

所以我们对于每一种右端点p,建立一颗主席树,然后在对应的主席树上高即可。

code:

 #pragma GCC optimize(2)
 #include <cstdio>
 #include <cstring>
 #include <algorithm>
 #include <map>
 #include <queue>
 #define LL long long
 #define mp make_pair
 #define pli pair <LL,int>

 using namespace std;

 ,M=;

 int n,m; map <int,int> pre;
 struct node {
     LL v; int x,l,r,p;
     node () {}
     node (LL _v,int _x,int _l,int _r,int _p) :
         v(_v),x(_x),l(_l),r(_r),p(_p) {}
     bool operator < (const node &o) const {
         return v<o.v;
     }
 };
 priority_queue <node> q;

 namespace TREE {
     int tot,rt[N],lc[M],rc[M]; pli w[M]; LL tag[M];
     #define mid (((l)+(r))>>1)
     #define ms(a,x) memset(a,x,sizeof a)
     inline ,ms(tag,);}
     inline void build (int &u,int l,int r) {
         w[u=++tot]=mp(,l);
         if (l==r) return;
         build(lc[u],l,mid),build(rc[u],mid+,r);
     }
     inline void insert (int &u,int v,LL z) {
         tag[u=++tot]=tag[v]+z;
         lc[u]=lc[v],rc[u]=rc[v],w[u]=w[v],w[u].first+=z;
     }
     inline void upload (int u) {
         w[u]=max(w[lc[u]],w[rc[u]]);
     }
     inline void download (int &u) {
         insert(lc[u],lc[u],tag[u]),insert(rc[u],rc[u],tag[u]);
         tag[u]=;
     }
     inline void modify (int &u,int v,int l,int r,int x,int y,LL z) {
         if (x<=l&&r<=y) {insert(u,v,z); return;}
         if (tag[u]) download(v);
         lc[u=++tot]=lc[v],rc[u]=rc[v],w[u]=w[v];
         if (x<=mid) modify(lc[u],lc[v],l,mid,x,y,z);
         ,r,x,y,z);
         upload(u);
     }
     inline pli query ()) {
         if (x<=l&&r<=y) return w[u];
         if (tag[u]) download(u);
         if (x<=mid) ret=query(lc[u],l,mid,x,y);
         ,r,x,y));
         return ret;
     }
 } using namespace TREE;

 inline void extend (int x,int l,int r) {
     if (l>r) return;
     pli nxt=query(x,,n,l,r);
     q.push(node(nxt.first,x,l,r,nxt.second));
 }

 int main () {
     scanf(],,n);
     ,x; i<=n; ++i) {
         scanf("%d",&x);
         modify(rt[i],rt[i-],,n,pre[x]+,i,(LL)x);
         extend(rt[i],,i),pre[x]=i;
     }
     node cur;
     for ( ; m; --m) {
         cur=q.top(),q.pop();
         extend(cur.x,cur.l,cur.p-);
         extend(cur.x,cur.p+,cur.r);
     }
     printf("%lld\n",cur.v);
     ;
 }

[bzoj P4504] K个串的更多相关文章

  1. bzoj : 4504: K个串 区间修改主席树

    4504: K个串 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 268  Solved: 110[Submit][Status][Discuss] ...

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

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

  3. BZOJ 4504: K个串

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

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

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

  5. bzoj4504 k个串 kstring 可持久化线段树 (标记永久化)

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

  6. 数据结构(主席树):COGS 2213. K个串

    2213. K个串 ★★★★   输入文件:bzoj_4504.in   输出文件:bzoj_4504.out   简单对比时间限制:20 s   内存限制:512 MB [题目描述] 兔子们在玩k个 ...

  7. BZOJ 3110 K大数查询 | 整体二分

    BZOJ 3110 K大数查询 题面 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c 如果是2 a b c形式,表示询问从第a个 ...

  8. 问题 K: 周期串plus

    问题 K: 周期串plus 时间限制: 1 Sec  内存限制: 128 MB提交: 682  解决: 237[提交] [状态] [命题人:外部导入] 题目描述 如果一个字符串可以由某个长度为k的字符 ...

  9. hiho#1449 重复旋律6 求长度为k的串最大次数 后缀自动机

    题目传送门 题目大意:求长度为k的串的最大次数,把k从1到length的所有答案全部输出. 思路: 这道题放在$SAM$里就是求长度$k$对应的所有$right$集中最大的大小. 我们以$aabab$ ...

随机推荐

  1. Requests卡死问题

    https://www.cnblogs.com/niansi/p/7143736.html https://blog.csdn.net/pilipala6868/article/details/807 ...

  2. MQ(队列消息的入门)

    消息中间件利用高效可靠的消息传递机制进行平台无关的数据交流,并基于数据通信来进行分布式系统的集成,通过提供消息传递和消息排队模型,它可以在分布式环境下拓展进程间的通信,对于消息中间件,常见的角色大致也 ...

  3. Bugku-CTF之Trim的日记本(不要一次就放弃)

    Day19 Trim的日记本 http://123.206.87.240:9002/ hints:不要一次就放弃

  4. Tomcat &servlet字符集编码问题

    1.字符编码的原由 1.1 request和response的默认编码是? 如果未指定字符编码,则Servlet规范要求使用ISO-8859-1的编码. HTTP消息正文(请求或响应)的字符编码在Co ...

  5. BZOJ-2298|区间dp|线段树

    problem a Description 一次考试共有n个人参加,第i个人说:"有ai个人分数比我高,bi个人分数比我低."问最少有几个人没有说真话(可能有相同的分数) Inpu ...

  6. P2178 [NOI2015]品酒大会

    思路 在后缀树上进行一些操作就好了 后缀树上LCA的maxlen就是两个后缀的LCP的长度了 然后统计每个点作为LCA的次数和最大值.次大值.最小值.次小值 然后就做完了 代码 #include &l ...

  7. git push时报错:Updates were rejected because the tip of your current branch is behind

    出现这样的问题是由于:自己当前版本低于远程仓库版本 有如下几种解决方法: 1.使用强制push的方法: git push -u origin master -f 这样会使远程修改丢失,一般是不可取的, ...

  8. Linux (麒麟)系统 重启后无法登陆进图形界面

    登录图形化界面的时候,会显示GNOME电源管理器没启动等提示信息,会一直卡在登录界面 在启动的时候按ESC或者在登录界面crtl+alt +f3 进入字符终端界面 查看物理存储空间占用信息,可能会有一 ...

  9. GeoJson

    几何对象.特征对象.特征对象集合.

  10. docker容器的时间同步

    好久没写博客了,有时间开始陆续整理一下工作中遇到的问题,今天罗列一下docker容器的时间同步问题 我们每次在run容器的时候,会存在时区不同的问题,这样对数据处理会有很大障碍,操作如下: 第一种方式 ...