\(\\\)

\(Description\)


给出一个数轴上\(N\)个点的坐标\(A_i\),选择\(K\)个点对,使得这\(K\)个点对每个点对的距离之和尽可能小。

  • \(N\in [0,10^5]\),\(K\in [0,\frac{N}{2}]\),\(A_i\in [0,10^9]\)

\(\\\)

\(Solution\)


  • 排序,求出相邻两两间距\(D_i\),问题转化为选\(K\)个互不相邻的间距
  • 将所有的间距放到一个小根堆中,每次取堆顶加入答案。如果选择了\(D_i\),则在堆中删去\(D_i,D_{i-1},D_{i+1}\),并将\(D_{i-1}+D_{i+1}-D_i\)加入堆中,去执行选\(K-1\)个值得子问题。
  • 这样做法的合理性在于,如果每次选则的是原来的\(D_i\),那么就是直接加入这个答案,如果选择的是某次操作后的\(D_{i-1}+D_{i+1}-D_i\),那么代表\(D_i\)在此前一定加入过答案,此时答案里累加上这个数代表,从答案中去掉原来选择的\(D_i\),加入\(D_i\)两侧的数,同样会使得选则的间距\(+1\)。
  • 注意,在合并边界元素时,如果再次进行反转操作并不会使选中的段数增加,所以如果边界被合并了一次重置该位置数字时应设为正无穷。
  • 可以发现几次操作以后前驱后继就很难寻找了,所以可以采用链表来维护这个数列,同时在插入堆中时绑定链表节点的编号,此时操作即改为合并三个链表元素,标记堆中三个元素不合法,执行\(K\)次即可选出合法的\(K\)段间距,即\(K\)个点对。

\(\\\)

\(Code\)


#include<cmath>
#include<queue>
#include<cstdio>
#include<cctype>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define R register
#define gc getchar
#define N 100010
#define inf 1000000000
using namespace std; inline int rd(){
int x=0; bool f=0; char c=gc();
while(!isdigit(c)){if(c=='-')f=1;c=gc();}
while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=gc();}
return f?-x:x;
} bool vis[N];
int n,m,ans,pre[N],nxt[N],num[N];
priority_queue<pair<int,int> > q; int main(){
n=rd(); m=rd();
for(R int i=2,lp=rd(),now;i<=n;++i){
pre[i]=i-1; nxt[i]=i+1;
num[i]=(now=rd())-lp; lp=now;
}
pre[2]=nxt[n]=0;
for(R int i=2;i<=n;++i) q.push(make_pair(-num[i],i));
while(m--){
while(vis[q.top().second]) q.pop();
int now=q.top().second;
int pr=pre[now],nx=nxt[now];
q.pop(); ans+=num[now];
pre[nxt[now]=nxt[nx]]=now;
nxt[pre[now]=pre[pr]]=now;
num[now]=(pr&&nx)?min(inf,num[pr]+num[nx]-num[now]):inf;
vis[pr]=vis[nx]=1; q.push(make_pair(-num[now],now));
}
printf("%d\n",ans);
return 0;
}

\(\\\)

种树就是把链表循环起来选最大,循环的关系不需要考虑边界的特判比上面的还水

\(\\\)

#include<cmath>
#include<queue>
#include<cstdio>
#include<cctype>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define R register
#define gc getchar
#define N 200010
#define inf 1000000000ll
using namespace std;
typedef long long ll; inline ll rd(){
ll x=0; bool f=0; char c=gc();
while(!isdigit(c)){if(c=='-')f=1;c=gc();}
while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=gc();}
return f?-x:x;
} bool vis[N];
ll n,m,ans,pre[N],nxt[N],num[N];
priority_queue<pair<int,int> > q; int main(){
n=rd(); m=rd();
if(n<m*2){puts("Error!");return 0;}
for(R ll i=1;i<=n;++i){
pre[i]=i-1; nxt[i]=i+1;
num[i]=rd();
}
pre[1]=n; nxt[n]=1;
for(R ll i=1;i<=n;++i) q.push(make_pair(num[i],i));
while(m--){
while(vis[q.top().second]) q.pop();
ll now=q.top().second;
ll pr=pre[now],nx=nxt[now];
q.pop(); ans+=num[now];
pre[nxt[now]=nxt[nx]]=now;
nxt[pre[now]=pre[pr]]=now;
num[now]=min(inf,num[pr]+num[nx]-num[now]);
vis[pr]=vis[nx]=1; q.push(make_pair(num[now],now));
}
printf("%lld\n",ans);
return 0;
}

[ CTSC 2007 / BZOJ 2151 ] Backup / 种树的更多相关文章

  1. [bzoj 2151]种树(贪心)

    题目:http://www.lydsy.com/JudgeOnline/problem.php?id=2151 分析:原型是bzoj 1150(CTSC 2007) 首先DP无法下手,想到贪心.想到贪 ...

  2. Guard Duty (medium) Codeforces - 958E2 || (bzoj 2151||洛谷P1792) 种树 || 编译优化

    https://codeforces.com/contest/958/problem/E2 首先求出N个时刻的N-1个间隔长度,问题就相当于在这些间隔中选K个数,相邻两个不能同时选,要求和最小 方法1 ...

  3. 洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心)

    洛谷P1484 种树&洛谷P3620 [APIO/CTSC 2007]数据备份 题解(堆+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/132 ...

  4. P3620 [APIO/CTSC 2007]数据备份

    P3620 [APIO/CTSC 2007]数据备份 题目描述 你在一家 IT 公司为大型写字楼或办公楼(offices)的计算机数据做备份.然而数据备份的工作是枯燥乏味的,因此你想设计一个系统让不同 ...

  5. 洛谷 P3620 [APIO/CTSC 2007]数据备份 解题报告

    P3620 [APIO/CTSC 2007]数据备份 题目描述 你在一家 IT 公司为大型写字楼或办公楼(offices)的计算机数据做备份.然而数据备份的工作是枯燥乏味的,因此你想设计一个系统让不同 ...

  6. 【BZOJ 2151】 2151: 种树 (贪心+堆+双向链表)

    2151: 种树 Description A城市有一个巨大的圆形广场,为了绿化环境和净化空气,市政府决定沿圆形广场外圈种一圈树.园林部门得到指令后,初步规划出n个种树的位置,顺时针编号1到n.并且每个 ...

  7. 【刷题】BZOJ 2151 种树

    Description A城市有一个巨大的圆形广场,为了绿化环境和净化空气,市政府决定沿圆形广场外圈种一圈树.园林部门得到指令后,初步规划出n个种树的位置,顺时针编号1到n.并且每个位置都有一个美观度 ...

  8. BZOJ 2151 种树(循环链表)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2151 [题目大意] 在一个长度为n的数字环中挑选m个不相邻的数字使得其和最大 [题解] ...

  9. bzoj 2151 种树——贪心+后悔

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2151 似乎是半年+前讲过的.(然而看到的时候却不会了) 考虑贪心,限制就是不能选两边的.如果 ...

随机推荐

  1. 【Codeforces 711C】Coloring Trees

    [链接] 我是链接,点我呀:) [题意] 连续相同的数字分为一段 你可以改变其中0为1~m中的某个数字(改变成不同数字需要不同花费) 问你最后如果要求分成恰好k段的话,最少需要多少花费 [题解] dp ...

  2. Ajax提交post请求返回404错误

    最近使用ajax提交表单的时候,发现无法执行success函数,后台的代码也正常执行了,但是就是无法执行success函数,执行error函数,返回的错误代码时404.显然是找不到请求的url. 可是 ...

  3. 【BZOJ4398】福慧双修(二进制,最短路)

    题意: 此题中S=1 思路:Orz ManGod秒切此题 我觉得出入边权互换不太直观,就改了一下写法 第一次默认与1有关的第一条出边只出不入,第二次默认只入不出 ..]of longint; head ...

  4. Sky数

    Problem Description Sky从小喜欢奇特的东西,而且天生对数字特别敏感,一次偶然的机会,他发现了一个有趣的四位数2992,这个数,它的十进制数表示,其四位数字之和为2+9+9+2=2 ...

  5. 2102 石子归并 2codevs

    2102 石子归并 2codevs 题目描述 Description 在一个园形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为 ...

  6. RAC 设置archive log模式

    首先设置 archive log的位置 SQL> alter system set log_archive_dest='+DATA/orcl/archive/'; System altered. ...

  7. Lua的面向对象程序设计

    Account={balance=} function Account.withdraw(self,v) self.balance=self.balance-v end a={balance=,wit ...

  8. Redis是单线程的

    Redis是单线程的 学习了: http://blog.csdn.net/liupeng_qwert/article/details/77263187 https://www.cnblogs.com/ ...

  9. Fedora下克隆Octopress博客

    我在自己的github上搭建了一个Octopress博客(http://songlee24.github.io/),用于不定期的写一些学习笔记和心得体会.但是有时候由于换了电脑或者重装了linux系统 ...

  10. Js通用验证

    //-----------------------------------------------------js 验证封装 zhy2014-07-10------------------------ ...