【题解】前k大子段和
题目描述
Peter喜欢玩数组。NOIP这天,他从Jason手里得到了一个大小为\(n\)的数组。
Peter求出了这个数组的所有子段和,并将这\(\frac{n(n+1)}{2}\)个数降序排列,他想知道前\(k\)个数是什么。
输入输出格式
输入格式
输入数据的第一行包含两个整数\(n\)和\(k\)。
接下来一行包含\(n\)个整数,代表数组。
输出格式
输出\(k\)个数,代表降序之后的前\(k\)个数,用空格隔开。
数据范围

题解
这个题目说的是十分的简洁明了,要求我们求出所有的子段和中前\(k\)大,首先看到这道题的时候,我用的二分答案加树状数组维护虽然在这道题上这种方法会T飞,但是,这种方法是一种方法是一种十分有效的算法。我们先二分答案来枚举第\(k\)大的子段和, 然后再用树状数组来维护和查询(就有点像求逆序对)。
具体过程:
我们每次枚举时,出第\(k\)大子段和为\(x\),那么\(x\)一定可以被表示为\(x = sum[i] - sum[j - 1]\)(\(sum[i]\)表示前缀和),我们把这个式子移项,可以得到\(sum[j - 1] = sum[i] - x\),这个式子表示当我们遍历到第 \(i\)个前缀和时,已知第\(k\)大的子段和为\(x\)那么我们只用找到\(sum[j - 1]\)之前有多少个\(sum[]\)就可以知道有多少个子段和比\(x\)大了。
#include<bits/stdc++.h>
using namespace std;
#define lowbit(x) ((x) & (-x))
const int MAX = 100005;
int n, k;
long long a[MAX], sum[MAX], tree[MAX];
vector <long long> s;
void Add(int x, long long val)
{
for(; x <= n; x += lowbit(x)) tree[x] += val;
}
long long Query(int x)
{
long long ret = 0;
for(;x ; x-= lowbit(x)) ret += tree[x];
return ret;
}
int check(int mid)
{
int ret = 0;
memset(tree, 0, sizeof(tree));
for(int i = 1; i <= n; ++ i)
{
int x = sum[i] - mid;
int it = lower_bound(s.begin(), s.end(), x) - s.begin();
ret += Query(it);
if(x > 0) ret ++;
it = lower_bound(s.begin(), s.end(), sum[i]) - s.begin();
Add(it + 1, 1);
}
return ret;
}
int main()
{
// freopen("ksum.in", "r", stdin);
// freopen("ksum.out", "w", stdout);
scanf("%d%d", &n, &k);
for(int i = 1; i <= n; ++ i)
{
scanf("%d", &a[i]);
sum[i] = sum[i - 1] + a[i];
s.push_back(sum[i]);
}
sort(s.begin(), s.end());
for(int t = 1; t <= k; ++ t)
{
long long l = 0, r = sum[n], mid, ans = 0;
for(;l < r;)
{
mid = (l + r) >> 1;
if(check(mid) >= t) l = mid + 1;
else r = mid;
}
printf("%lld ", l);
}
return 0;
}
如果求取的次数比较少的话,这也会是一个优秀的算法,但是,这道题的\(k\)太大,导致要多次重复这个过程所以,我们要考虑其他解法,因为数组中的数都是非负数,所以,我们可以来贪心。
- 最大的一定是所有数之和。
- 每次将最大的去头或去尾可以构成备选答案。
所以我们可以用优先队列来维护,但是,对于\([x,y]\)来说,它可能在\([x - 1, y]\)去头时加入,也有可能在\([x,y + 1]\)去尾时加入,这样就会重复,所以,我们需要一种不会重复的枚举方式,我们把所有前缀和入队,然后每次只考虑去头而不考虑去尾(在前缀和入队时已经去过了),这样就不会重复了。
#include<bits/stdc++.h>
using namespace std;
struct Node{
int l, r;
long long sum;
bool friend operator < (const Node & x, const Node & y)
{
return x.sum < y.sum;
}
};
priority_queue <Node> q;
long long a[100005], sum = 0;
Node make_Node(int l, int r, long long sum)
{
Node x;
x.l = l, x.r = r, x.sum = sum;
return x;
}
int main()
{
int n, k;
scanf("%d%d", &n, &k);
for(int i = 1; i <= n; ++ i)
{
scanf("%d", &a[i]);
sum += a[i];
q.push(make_Node(1, i, sum));
}
for(int i = 1; i <= k; ++ i)
{
Node x;
x = q.top(); q.pop();
printf("%lld ", x.sum);
q.push(make_Node(x.l + 1, x.r, x.sum - a[x.l]));
}
printf("\n");
return 0;
}
【题解】前k大子段和的更多相关文章
- HDU 6041 I Curse Myself(点双联通加集合合并求前K大) 2017多校第一场
题意: 给出一个仙人掌图,然后求他的前K小生成树. 思路: 先给出官方题解 由于图是一个仙人掌,所以显然对于图上的每一个环都需要从环上取出一条边删掉.所以问题就变为有 M 个集合,每个集合里面都有一堆 ...
- 前k大金币(动态规划,递推)
/* ///题解写的很认真,如果您觉得还行的话可以顶一下或者评论一下吗? 思路: 这题复杂在要取前k大的结果,如果只是取最大情况下的金币和,直接 动态规划递归就可以,可是前k大并不能找出什么公式,所以 ...
- 寻找无序数组中的前k大元素
题目描述 以尽可能小的代价返回某无序系列中的两个最大值,当有重复的时设置某种机制进行选择. 题解 首先要考虑的是重复的数的问题. A.不处理重复数据方法:在处理第k大的元素时不处理重复的数据,也就是将 ...
- 7617:输出前k大的数
7617:输出前k大的数 查看 提交 统计 提问 总时间限制: 10000ms 单个测试点时间限制: 1000ms 内存限制: 65536kB 描述 给定一个数组,统计前k大的数并且把这k个数从大到小 ...
- 输出前 k 大的数
总时间限制: 10000ms 单个测试点时间限制: 1000ms 内存限制: 65536kB 描述 给定一个数组,统计前k大的数并且把这k个数从大到小输出. 输入 第一行包含一个整数n,表示数组的大小 ...
- 牛客第六场 J.Heritage of skywalkert(On求前k大)
题目传送门:https://www.nowcoder.com/acm/contest/144/J 题意:给一个function,构造n个数,求出其中任意两个的lcm的最大值. 分析:要求最大的lcm, ...
- (算法)前K大的和
题目: 1.有两个数组A和B,每个数组有k个数,从两个数组中各取一个数加起来可以组成k*k个和,求这些和中的前k大. 2.有N个数组,每个数组有k个数,从N个数组中各取一个数加起来可以组成k^N个和, ...
- 快速排序算法的实现 && 随机生成区间里的数 && O(n)找第k小 && O(nlogk)找前k大
思路:固定一个数,把这个数放到合法的位置,然后左边的数都是比它小,右边的数都是比它大 固定权值选的是第一个数,或者一个随机数 因为固定的是左端点,所以一开始需要在右端点开始,找一个小于权值的数,从左端 ...
- 输出前k大的数
总时间限制: 10000ms单个测试点时间限制:1000ms内存限制:65536kB(noi) 描述 给定一个数组,统计前k大的数并且把这k个数从大到小输出. 输入 第一行包含一个整数n,表示数组的大 ...
随机推荐
- Python——基本的方法
格式化 我们经常会输出类似'亲爱的xxx你好!你xx月的话费是xx,余额是xx'之类的字符串,而xxx的内容都是根据变量变化的,所以,需要一种简便的格式化字符串的方式 >>> 'He ...
- 分布式ID生成器PHP+Swoole实现(上) - 实现原理
1.发号器介绍 什么是发号器? 全局唯一ID生成器,主要用于分库分表唯一ID,分布式系统数据的唯一标识. 是否需要发号器? 1)是否需要全局唯一. 分布式系统应该不受单点递增ID限制,中心式的会涉及到 ...
- Rafy中的EventBus
EventBus主要是干嘛使的,直接翻译叫事件总线. 是观察者模型的实现,利用它你既可以实现观察者模型的业务场景,还可以基于它的事件驱动机制来实现应用程序内组件之间的解耦与通信. 我们来看看有Even ...
- HBase入门教程
# 背景 最近看到公司一个项目用到hbase, 之前也一直想看下hbase.个人理解Hbase作为一个nosql数据库,逻辑模型感觉跟关系型数据库有点类似.一个table,有row即行,列.不过列是一 ...
- 关于JAVA项目报表选型过程
本人一直在走.NET技术路线,考虑到后期公司搞JAVA项目,也算是进行技术灾备,开始对JAVA技术进行关注.万事开头难,也是上来一头包.没办法,顶着上吧.上面开始分给我任务了.就是对后期项目报表进行方 ...
- [翻译]Review——Learn these core JavaScript concepts in just a few minutes
Learn these core JavaScript concepts in just a few minutes(只需几分钟即可学习这些核心JavaScript概念) 原文地址:https://m ...
- mybatis 关联表心得
1,例如订单表与用户表的关联,一个订单对应一个用户,这是一对一关联: 用代码实现是这样: A(用resultType去实现的话,如下) 1,使用到继承, OrderUser extend Order{ ...
- linux shell实现守护进程 看门狗 脚本
嵌入式初学者,第一次上传代码.昨天做了一个udhcpd与udhcpc的守护,目前只会用shell模仿编写,还有什么方法可以做守护呢? ? 1 2 3 4 5 6 7 8 9 10 11 12 13 1 ...
- 媒体查询hack
随着Responsive设计的流行,Medial Queries可算是越来越让人观注了.他可以让Web前端工程实现不同设备下的样式选择,让站点在不同的设备中实现不同的效果.这个早前在 w3cplus已 ...
- 1 win10双系统安装ubuntu16.04和18.04(问题详细记录)
我从来不想在琐事上花功夫.就拿装系统来说,我感觉拿过来一个完全陌生的项目源码,看起来,都比装系统爽.我属于典型的逻辑思考男.喜欢畅游程序的海洋. 一直windows跑深度学习和tensorflow,有 ...