这道题是昨天linkfqy dalao上课讲的一道题

当时他讲的时候就想到了一种玄学的搞法,然后不敢相信自己切掉了

没想到后来CHJ dalao也想到了这种算法,然后发现是对的

后来10min就切掉了

看来以后要活跃一点了

这里主要讲两种算法(想WQS二分这种高深的算法就不讲了)

首先讲题意:一个人它只能在某个时间点去跟看k个他的客人对话,或是结束他和当前这个客人的对话。但是它不能在两个连续的时间段进行讲话,问它最少需要多少时间来和k个客人对话。

1.堆

我们先按时间排序,处理出每一个时间段的长度

然后就建模转化成:n-1个点中取k个并且每两个都不相邻的最少代价和。

这就和一道题很像了,看Luogu P1484&sol

用堆水一水即可

CODE

#include<cstdio>
#include<algorithm>
#include<queue>
using namespace std;
const int N=500005;
int k,n,x[N],a[N],pre[N],nxt[N],ans;
bool vis[N];
struct data
{
int x,num;
bool operator <(const data &s) const
{
return s.x<x;
}
};
priority_queue<data> small;
inline char tc(void)
{
static char fl[100000],*A=fl,*B=fl;
return A==B&&(B=(A=fl)+fread(fl,1,100000,stdin),A==B)?EOF:*A++;
}
inline void read(int &x)
{
x=0; char ch=tc();
while (ch<'0'||ch>'9') ch=tc();
while (ch>='0'&&ch<='9') x=x*10+ch-'0',ch=tc();
}
int main()
{
//freopen("CODE.in","r",stdin); freopen("CODE.out","w",stdout);
register int i; read(k); read(n);
for (i=1;i<=n;++i)
read(x[i]);
sort(x+1,x+n+1);
for (i=2;i<=n;++i)
{
a[i-1]=x[i]-x[i-1];
pre[i-1]=i-2; nxt[i-1]=i;
small.push((data){a[i-1],i-1});
}
a[0]=a[n]=1e9; pre[n]=n-1; nxt[0]=1;
while (k--)
{
while (vis[small.top().num]) small.pop();
data p=small.top(); small.pop(); ans+=p.x;
p.x=a[p.num]=a[pre[p.num]]+a[nxt[p.num]]-a[p.num];
vis[pre[p.num]]=vis[nxt[p.num]]=1;
pre[p.num]=pre[pre[p.num]]; nxt[pre[p.num]]=p.num;
nxt[p.num]=nxt[nxt[p.num]]; pre[nxt[p.num]]=p.num;
small.push(p);
}
printf("%d",ans);
return 0;
}

2.DP

首先这种题目就是看一眼就想到DP的吧

我们令f[i][j][0/1]表示前i个点中取j个并且第i个点取or不取(1表示取,2表示不取)的最少代价

则很容易写出转移:

  • f[i][j][1]=f[i-1][j-1][0]+a[i]

  • f[i][j][0]=min(f[i-1][j][0],f[i-1][j][1])

首先我们发现i的那一维可以滚动存储优化掉

然后空间就够了,但是时间为O(nk)

然后坚信CF的机子是无敌的满怀信仰地交上去说不定就过了

然后这里我们就要讲一个结论了:

无论怎么取,都只可能取到3*k小的数

Why?很简单,如果所有的数左右都取,算上它自己,也就3个,扩展开就是3*k

所以我们就把时间优化到了O(k^2)

CODE

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=500005,K=5005;
int x[N],a[N],n,k,f[2][K][2],pre,num,tot;
inline char tc(void)
{
static char fl[100000],*A=fl,*B=fl;
return A==B&&(B=(A=fl)+fread(fl,1,100000,stdin),A==B)?EOF:*A++;
}
inline void read(int &x)
{
x=0; char ch=tc();
while (ch<'0'||ch>'9') ch=tc();
while (ch>='0'&&ch<='9') x=x*10+ch-'0',ch=tc();
}
inline int min(int a,int b)
{
return a<b?a:b;
}
int main()
{
//freopen("CODE.in","r",stdin); freopen("CODE.out","w",stdout);
register int i,j;
read(k); read(n);
for (i=1;i<=n;++i)
read(x[i]); sort(x+1,x+n+1);
for (i=2;i<=n;++i)
x[i-1]=a[i-1]=x[i]-x[i-1];
sort(x+1,x+n); num=3*k<n?x[3*k]:x[n-1];
memset(f[0],63,sizeof(f[0])); f[0][0][0]=f[0][0][1]=0;
int now=0,lst=1,pre=-1;
for (i=1;i<n;++i)
{
if (a[i]>num) continue;
now^=1; lst^=1; memset(f[now],63,sizeof(f[now]));
++tot;
if (i-1!=pre)
{
for (j=1;j<=min(tot,k);++j)
{
f[now][0][0]=f[now][0][1]=0;
f[now][j][1]=min(f[lst][j-1][0],f[lst][j-1][1])+a[i];
f[now][j][0]=min(f[lst][j][0],f[lst][j][1]);
}
} else
{
for (j=1;j<=min(tot,k);++j)
{
f[now][0][0]=f[now][0][1]=0;
f[now][j][1]=f[lst][j-1][0]+a[i];
f[now][j][0]=min(f[lst][j][0],f[lst][j][1]);
}
} pre=i;
}
printf("%d",min(f[tot&1][k][0],f[tot&1][k][1]));
return 0;
}

CF 958E2. Guard Duty (medium)的更多相关文章

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

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

  2. [Codeforces958E2]Guard Duty (medium)(区间DP)

    Description 题目链接 Solution 可以把题目转化一下模型,将间隔取出来,转化为N-1个数,限制不能取相邻两个数,求取K个数的最小价值 设DP[i][j]表示前i个数取j个最大价值(第 ...

  3. CF 85E Guard Towers——二分图染色

    题目:http://codeforces.com/contest/85/problem/E 当然是二分.然后连一个图,染色判断是不是二分图即可.方案数就是2^(连通块个数). 别真的连边!不然时间空间 ...

  4. Guard Duty (hard) Codeforces - 958E3 || uva 1411

    https://codeforces.com/contest/958/problem/E3 当没有三点共线时,任意一个这样的点集都是保证可以找到答案的,(考虑任意一种有相交的连线方案,一定可以将其中两 ...

  5. CF E2 - Daleks' Invasion (medium) (LCA求两点树上路径上的最大边权)

    http://codeforces.com/contest/1184/problem/E2 题意:给出一副图,首先求出这幅图的最小生成树 , 然后修改这幅图上不属于最小生成树的边权,使得修改后的图在求 ...

  6. 决策单调性&wqs二分

    其实是一个还算 trivial 的知识点吧--早在 2019 年我就接触过了,然鹅当时由于没认真学并没有把自己学懂,故今复学之( 1. 决策单调性 引入:在求解 DP 问题的过程中我们常常遇到这样的问 ...

  7. DP 优化方法大杂烩 & 做题记录 I.

    标 * 的是推荐阅读的部分 / 做的题目. 1. 动态 DP(DDP)算法简介 动态动态规划. 以 P4719 为例讲一讲 ddp: 1.1. 树剖解法 如果没有修改操作,那么可以设计出 DP 方案 ...

  8. 20190708三人开黑CF模拟赛

    7月8号晚上8点和两位巨佬开了一场虚拟cf: [Helvetic Coding Contest 2018 online mirror (teams allowed, unrated)] 我这么蔡,只A ...

  9. 越狱Season 1- Episode 16

    Season 1, Episode 16 -Burrows:Don't be. It's not your fault. 不要,不是你的错 -Fernando: Know what I like? 知 ...

随机推荐

  1. iOS开发-本地存储(偏好设置,Plist,归档)

    1.   NSUserDefaults //TODO: 1.NSUserDefaults NSUserDefaults类除了可以存储数组.字典.NSdata外,还可以直接存储OC基本类型属性.但是不能 ...

  2. [Kali_Debian] 清除无用的库文件(清理系统,洁癖专用)-布布扣-bubuko.com - Google Chrome

    [Kali_Debian] 清除无用的库文件(清理系统,洁癖专用) 时间:2014-11-07 16:25:36      阅读:1486      评论:0      收藏:0      [点我收藏 ...

  3. python第三十三天----静态方法、类方法、属性方法

    @staticmethod 装饰后,类中的方法转成静态方法 class a: @staticmethod def b(self): print('') 静态方法不可以访问实例变量或类变量,相当于类中的 ...

  4. sqlserver 拆分

    有表tb, 如下:id value----------- -----------1 aa,bb2 aaa,bbb,ccc欲按id,分拆value列, 分拆后结果如下:id value--------- ...

  5. VMware虚拟机中CentOS 7的硬盘空间扩容

    查看centos7系统挂载点信息 扩展VMWare-centos7硬盘空间 对新增加的硬盘进行分区.格式化 添加新LVM到已有的LVM组,实现扩容 1.查看centos7系统挂载点信息 df -h查看 ...

  6. PHP LAMP环境搭建及网站配置流程(完整版)

    心血来潮想做一个自己的博客网站,写一些文章做技术分享,平时遇到的一些问题的解决办法都记录下来,网站搭建成功,那么第一篇博客自然就是整个网站的搭建以及域名的注册.备案.解析流程,总共分为以下几步: 1. ...

  7. 使用Maven分环境打包:dev sit uat prod

    使用Maven管理的项目,经常需要根据不同的环境打不同的包,因为环境不同,所需要的配置文件不同,比如database的连接信息,相关属性等等. 在Maven中,我们可以通过P参数和profiles元素 ...

  8. Tomcat 访问页面或服务器异常,请检查这些方面

    若还没有部署网站,请检查 防火墙是否关闭 数据库服务是否打开 浏览器访问的地址和端口是否正确 tomcat 配置文件中的端口是否发生冲突,换一个试试 若出现的是"拒绝连接",检查阿 ...

  9. 解决jqueryeasyUI dialog 弹出窗体超出浏览器,导致不能关闭的bug

    使用panel的onMove事件攻克了panel,dialog以及window组件在被拖动时,会超出浏览器边界而无法拖回的情况. 当窗体被拖出浏览器有边界时.$(document).width();会 ...

  10. Android平台调用Web Service:线程返回值

    接上文 前文中的遗留问题 对于Java多线程的理解,我曾经只局限于实现Runnable接口或者继承Thread类.然后重写run()方法.最后start()调用就算完事,可是一旦涉及死锁以及对共享资源 ...