K个串
题目链接
题解
看完题目后可以立刻想到:先算出最大值, 然后把最大值剔除掉,再找此时的最大值也就是次大值。这样重复\(k\)边即可找到第\(k\)大值。
于是我们只需要考虑找最大值了
我们可以维护后缀和中的最大值(这里的和是指题目中的不统计重复数字的求和)
具体来说, 我们可以建\(n\)课线段树, 第\(i\)颗存的是以\(i\)为结尾的所有后缀和, 那么, 我们可以把每颗线段树的最大值全部扔进一个大根堆, 这样我们就能每次得到当前的最大值。
接着, 我们从堆中取出当前最大值, 设这个最大值是在第\(i\)棵树的\([l, r]\)区间内取得的, 位置为\(p\), 那么我们分别在第\(i\)棵树的\([l, p)\)和\((p, r]\)两区间内找最大值, 然后把它们放入堆中。
显然, 我们可以用主席树优化空间\((\)pushdown一定要记得复制节点!!!!!\()\)
注意: 由于不统计重复数字, 那么在建主席树时, 设当前数字\(a_i\)上次出现位置为\(l\), 那么它的贡献区间为\((p, i)\)
代码
#include <iostream>
#include <cstring>
#include <cstdio>
#include <cstdlib>
#include <algorithm>
#include <vector>
#include <queue>
#include <map>
using namespace std;
typedef long long LL;
#define fi first
#define se second
typedef pair <LL, int> pr;
const LL INF = 1e14;
const int N = 100010;
int root[N];
struct SegmentTree
{
int ls[N * 400], rs[N * 400], sz;
LL addv[N * 400];
pr maxv[N * 400];
int cpyNode(int x)
{
int cur = ++sz;
ls[cur] = ls[x];
rs[cur] = rs[x];
maxv[cur] = maxv[x];
addv[cur] = addv[x];
return cur;
}
inline int downtag(int x, LL v)
{
int cur = cpyNode(x);
addv[cur] += v;
maxv[cur].fi += v;
return cur;
}
inline void pushdown(int cur)
{
if (addv[cur])
{
if (ls[cur]) ls[cur] = downtag(ls[cur], addv[cur]);
if (rs[cur]) rs[cur] = downtag(rs[cur], addv[cur]);
addv[cur] = 0;
}
}
void build(int & cur, int l, int r)
{
cur = ++sz;
addv[cur] = 0;
if (l == r) { maxv[cur] = make_pair(0, -l); return; }
int mid = (l + r) >> 1;
build(ls[cur], l, mid);
build(rs[cur], mid+1, r);
maxv[cur] = max(maxv[ls[cur]], maxv[rs[cur]]);
}
void update(int & cur, int pre, int l, int r, int ql, int qr, LL v)
{
if (ql == l && qr == r)
cur = downtag(pre, v);
else
{
pushdown(pre);
cur = cpyNode(pre);
int mid = (l + r) >> 1;
if (qr <= mid)
update(ls[cur], ls[pre], l, mid, ql, qr, v);
else if (ql > mid)
update(rs[cur], rs[pre], mid+1, r, ql, qr, v);
else
update(ls[cur], ls[pre], l, mid, ql, mid, v),
update(rs[cur], rs[pre], mid+1, r, mid+1, qr, v);
maxv[cur] = max(maxv[ls[cur]], maxv[rs[cur]]);
}
}
pr query(int cur, int l, int r, int ql, int qr)
{
if (ql == l && qr == r)
return maxv[cur];
pushdown(cur);
int mid = (l + r) >> 1;
if (qr <= mid)
return query(ls[cur], l, mid, ql, qr);
else if (ql > mid)
return query(rs[cur], mid+1, r, ql, qr);
else
return max(query(ls[cur], l, mid, ql, mid),
query(rs[cur], mid+1, r, mid+1, qr));
}
} seg;
int n, k;
LL a[N]; int last[N];
map <LL, int> pos;
struct Data
{
int id, pl, pr, pos; LL val;
Data() { }
Data(int _1, int _2, int _3, int _4, LL _5) : id(_1), pl(_2), pr(_3), pos(_4), val(_5) { }
bool operator < (const Data & rhs) const { return val < rhs.val; }
};
priority_queue <Data> q;
int main()
{
scanf("%d %d", &n, &k);
for (int i = 1; i <= n; i++)
{
scanf("%lld", &a[i]);
last[i] = pos[a[i]];
pos[a[i]] = i;
}
seg.build(root[0], 1, n);
for (int i = 1; i <= n; i++)
seg.update(root[i], root[i-1], 1, n, last[i]+1, i, a[i]);
for (int i = 1; i <= n; i++)
{
pr res = seg.query(root[i], 1, n, 1, i);
q.push(Data(i, 1, i, -res.se, res.fi));
}
Data res;
for (int i = 1; i <= k; i++)
{
res = q.top(); q.pop();
if (res.pl < res.pos)
{
pr now = seg.query(root[res.id], 1, n, res.pl, res.pos-1);
q.push(Data(res.id, res.pl, res.pos-1, -now.se, now.fi));
}
if (res.pr > res.pos)
{
pr now = seg.query(root[res.id], 1, n, res.pos+1, res.pr);
q.push(Data(res.id, res.pos+1, res.pr, -now.se, now.fi));
}
}
printf("%lld\n", res.val);
return 0;
}
K个串的更多相关文章
- 数据结构(主席树):COGS 2213. K个串
2213. K个串 ★★★★ 输入文件:bzoj_4504.in 输出文件:bzoj_4504.out 简单对比时间限制:20 s 内存限制:512 MB [题目描述] 兔子们在玩k个 ...
- [bzoj P4504] K个串
[bzoj P4504] K个串 [题目描述] 兔子们在玩k个串的游戏.首先,它们拿出了一个长度为n的数字序列,选出其中的一个连续子串,然后统计其子串中所有数字之和(注意这里重复出现的数字只被统计一次 ...
- bzoj : 4504: K个串 区间修改主席树
4504: K个串 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 268 Solved: 110[Submit][Status][Discuss] ...
- 问题 K: 周期串plus
问题 K: 周期串plus 时间限制: 1 Sec 内存限制: 128 MB提交: 682 解决: 237[提交] [状态] [命题人:外部导入] 题目描述 如果一个字符串可以由某个长度为k的字符 ...
- hiho#1449 重复旋律6 求长度为k的串最大次数 后缀自动机
题目传送门 题目大意:求长度为k的串的最大次数,把k从1到length的所有答案全部输出. 思路: 这道题放在$SAM$里就是求长度$k$对应的所有$right$集中最大的大小. 我们以$aabab$ ...
- 【BZOJ4504】K个串 可持久化线段树+堆
[BZOJ4504]K个串 Description 兔子们在玩k个串的游戏.首先,它们拿出了一个长度为n的数字序列,选出其中的一个连续子串,然后统计其子串中所有数字之和(注意这里重复出现的数字只被统计 ...
- bzoj 4504: K个串 可持久化线段树+堆
题目: Description 兔子们在玩k个串的游戏.首先,它们拿出了一个长度为n的数字序列,选出其中的一 个连续子串,然后统计其子串中所有数字之和(注意这里重复出现的数字只被统计一次). 兔子们想 ...
- bzoj4504 k个串 kstring 可持久化线段树 (标记永久化)
[fjwc2015]k个串 kstring [题目描述] 兔子们在玩k个串的游戏.首先,它们拿出了一个长度为n的数字序列,选出其中的一个连续子串,然后统计其子串中所有数字之和(注意这里重复出现的数字只 ...
- spoj 7258 SUBLEX(求第k大字串
其实对sam的拓扑排序我似懂非懂但是会用一点了. /** @xigua */ #include <stdio.h> #include <cmath> #include < ...
- bzoj 2998 第k小字串
这道题用后缀数组貌似会T. 后缀自动机做法: t==0:第k小的本质不同字串 首先把后缀自动机建出来,我们会得到一个DAG,并且只存在一个点入度为0(我们称之为根),可以证明字符串的任意一个本质不同的 ...
随机推荐
- SpringBoot系列:二、SpringBoot的配置文件
SpringBoot的配置文件在resources文件夹下 springboot的配置文件支持两种形式的写法,一种是经典的properties另一种是yml yml通过空格缩进的形式来表示对象的层级关 ...
- laravel使用artisan报错SQLSTATE[42S02]: Base table or view not found: 1146
说明你在应用初始化阶段使用到了数据库层面的东西,然而当时数据库不存在这个表/字段,所以会报错 需要在初始化比如 config 目录配置中,使用了数据库,在使用前需要添加一层判断,如果不存在 你需要用到 ...
- 观察者模式(jdk实现)
1.定义 在对象中定义一对多的依赖,当一个对象改变状态,依赖它的对象会收到通知并更新. 2.实现 (主要通过jdk自己定义的观察者实现) 以气象站通知展示板为例子,当气象站收到的各种参数改变的时候 ...
- phpmyadmin和网页上面的乱码问题
前段时间做了个留言板,但是总是出现乱码,而且出现了无法插入的情况:发现这么个问题:在phpmyadmin里面默认的编码是瑞士的一个编码:我就郁闷了为什么这么一个通用软件的编码放着UTF-8或者是GBK ...
- win10编写8086汇编程序(dosbox)
有部分同学反馈.在使用edit命令来编写汇编程序时遇到问题,由于模拟器没有edit程序,所以要换一种方式编写源程序.下面是完整的演示. 视频链接:http://www.bilibili.com/vid ...
- PHPstorm Xdebugger最全详细
0 Xdebug调试的原理(选看) 图0-1 单机调试原理示意图 图0-2 多机调试原理示意图 对于PHP开发,初来咋到,开发环境的搭建和理解感觉是最烦人的一件事了.不像JAVA,打开一个Eclips ...
- JSP程序不能正常运行 MyEclipse10 Tomcat6.0
我写的sp程序,上午运行正常:但是下午再打开运行会提示对jsp解释失败 谁知道这是怎么回事呢? 后来是发现: 要运行JSP程序 Myeclipse10和Tomcat6的jdk都要调整到jdk1.7的版 ...
- 腾讯视频的手机端的地址和PC端的地址是不一样的
腾讯视频的手机端的地址和PC端的地址是不一样的,所以使用iframe的时候记得要使用手机端的地址
- react 之 flux
[WangQi]---flux---[react] 一.什么是Flux Flux 是一种架构思想,专门解决软件的结构问题.它跟MVC 架构是同一类东西,但是更加简单和清晰. 二.flux的基本概念 ...
- Redis 21问,你接得住不?
作者:菜鸟小于 cnblogs.com/Young111/p/11518346.html 1.什么是redis? Redis 是一个基于内存的高性能key-value数据库. 2.Reids的特点 R ...