hihocoeder1384

算法竞赛进阶指南上的题目

我们肯定是吧最大值和最小值匹配,次大值和次小值匹配以此类推

首先,类似于区间覆盖的思想,我们对于一个\(L\),找到最大的满足条件的\(R\)

之后把\(R + 1\)作为下一个\(L\)继续这个操作

现在,问题转化成了我们如何寻找最大的\(R\)

一个比较明显的思路就是去二分,但是二分时间复杂度不对

因为如果每次只能前进一格,二分时间复杂度就变成了\(n^2\log{n}\)

考虑倍增的思想

我们对于每个\(L\)

初始设置\(R= L - 1,p = 1\)

之后求加上\([R + 1,R + p]\)时候合法

合法的话 \(p = p \times 2\)

否则 \(p = p / 2\)

每次暴力判断新加入的区间是否合法

可以证明这样时间复杂度

为\(n\log^2{n}\)

之后发现,对于新加进来的区间,我们只需要把它排序,然后和原有的区间进行归并即可

时间复杂度就变成了\(n\log n\)

#include<cstdio>
#include<iostream>
#include<queue>
#include<algorithm>
#include<cstring>
#include<cctype>
#include<vector>
#include<ctime>
#include<cmath>
#define LL long long
#define pii pair<int,int>
#define mk make_pair
#define fi first
#define se second
#define min std::min
#define max std::max
const int N = 5e5 + 3;
int a[N];
int n,m;LL k;
std::vector <int> A,B,G;
inline LL read(){
LL v = 0,c = 1;char ch = getchar();
while(!isdigit(ch)){
if(ch == '-') c = -1;
ch = getchar();
}
while(isdigit(ch)){
v = v * 10 + ch - 48;
ch = getchar();
}
return v * c;
}
inline bool check(int l,int r){
B.clear(),G.clear();
for(int i = l;i <= r;++i) B.push_back(a[i]);
sort(B.begin(),B.end());
int now1 = 0;int now2 = 0;
while(now1 < A.size() && now2 < B.size()){
if(A[now1] < B[now2]) G.push_back(A[now1++]);
else G.push_back(B[now2++]);
}
while(now1 < A.size()) G.push_back(A[now1++]);
while(now2 < B.size()) G.push_back(B[now2++]);
LL res = 0;
now1 = 0,now2 = G.size() - 1;
for(int cnt = 0;cnt < m && now1 < now2;now1++,now2--,cnt++) res += 1ll * (G[now1] - G[now2]) * (G[now1] - G[now2]);
if(res <= k){
A = G;
return 1;
}
else return 0;
}
int main(){
// freopen("A.in","r",stdin);
// freopen("A1.out","w",stdout);
int T = read();
while(T--){
int ans = 0;
n = read(),m = read(),k = read();
for(int i = 1;i <= n;++i) a[i] = read();
int l = 1,r = 0;
while(l <= n){
int p = 1;
A.clear();
while(p){
int t = min(n,r + p);
if(check(r + 1,t)){
r = t;
if(t == n) break;
p <<= 1;
}
else p >>= 1;
}
l = r + 1;
ans++;
}
printf("%d\n",ans);
}
return 0;
}
/*
1
10 2 10
4 1 3 2 1 3 2 1 3 4 */

hihocoeder1384的更多相关文章

随机推荐

  1. WPF/Silverlight深度解决方案:(七)HLSL自定义渲染特效之完美攻略(中)

    原文:WPF/Silverlight深度解决方案:(七)HLSL自定义渲染特效之完美攻略(中) 通过上一节的解说,大家是否已经对HLSL有了较深刻的认识和理解,HLSL的渲染不仅仅局限于静态处理,通过 ...

  2. jdbc框架-dbutils的简单使用

    jdbc框架-dbutils的简单使用 dbutils:是apache组织的一个工具类,jdbc的框架,更方便我们使用 使用步骤: 1.导入jar包(commons-dbutils-1.4.jar) ...

  3. @hdu - 6687@ Rikka with Stable Marriage

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 给定一个稳定婚姻匹配问题,其中第 i 个男生与第 j 个女生之间 ...

  4. python == 符号

  5. python print函数

  6. Redis源码解析:02链表

    链表提供了高效的节点重排能力,以及顺序性的节点访问方式,因为Redis使用的C语言并没有内置这种数据结构,所以Redis自己实现了链表. 链表在Redis中的应用非常广泛,比如列表的底层实现之一就是链 ...

  7. Android教程 -05 Android6.0权限的管理

    视频为本篇博客知识的讲解,建议采用超清模式观看, 欢迎点击订阅我的优酷 上篇文章我们讲解了通过隐式意图拨打电话,在AndroidManifest.xml文件中添加了权限 <uses-permis ...

  8. Rikka with Mista 线段树求交点个数

    由于上下线段是不可能有交点的 可以先看左右线段树,按照y递增的顺序,对点进行排序. 升序构造,那么对于从某一点往下的射线,对于L,R进行区间覆盖,线段交点个数就是单点的被覆盖的次数. 降序构造,那么对 ...

  9. H3C 局域网的不足

  10. phpStudy本地环境测试,打开网页很慢的解决办法!

    很多人应该都遇到了在使用phpStudy本地环境测试软件时候打开很慢的问题,甚至动辄达到了1000ms以上,开篇直接给出解决办法: 下面给大家介绍phpstudy访问速度慢的解决办法. 1.修改mys ...