Luogu 5017 NOIP2018普及组T3 摆渡车 (斜率优化 + 必要的转移进行剪枝)
题意:
有 n 名同学要乘坐摆渡车从人大附中前往人民大学,第 i 位同学在第 ti 分钟去 等车。只有一辆摆渡车在工作,但摆渡车容量可以视为无限大。摆渡车从人大附中出发、 把车上的同学送到人民大学、再回到人大附中(去接其他同学),这样往返一趟总共花费 m 分钟(同学上下车时间忽略不计)。摆渡车要将所有同学都送到人民大学。
凯凯很好奇,如果他能任意安排摆渡车出发的时间,那么这些同学的等车时间之和最小为多少呢?
本人感觉自己语文不是那么的好,无法概括。
细节:
咳咳,此题细节都在题意之中嘻嘻嘻,原来懒还有这个好处。
分析:
round1:从时间入手状态迎刃而解:f[i] 表示以 i 作为发车时间的最小等车时间之和。
则转移就为:f[i] = min{ f[j] + Σj<tk≤i i-tk }
但是这个转移不免需要循环计算求和的内容,我们需要考虑将其利用 O(1) 进行计算完毕,考虑将这个式子进行打开:
Σj<tk≤i i-tk = i×Σj<tk≤i 1 - Σj<tk≤i tk,不难发现此时我们只需要预处理出 tk 出现位置的前缀数组 cnt[i],以及 tk 出现位置的值的前缀和 sum[i]。
这样方程进行进一步转化:f[i] = min{ f[j] + ( cnt[i] - cnt[j] ) × i - ( sum[i] - sum[j] ) }
最后观察其是否存在单调性,显然上方的 1D/1D 的动态规划需要进行斜率优化了,裸体希望有这些意识吧,常规套路,最后只需要维护一个斜率:
k = ( f[j2] + sum[j2] - f[j1] - sum[j1] ) / (cnt[j2] - cnt[j1]) 单调递增即可,具体方法详见本人的第一篇博客。
round2:确实从时间入手是一种思路,但是对于较小的 n 和 m 很多选手应该也会有想法,不难发现对于第 i 个人对应的等待时间 ti 它的上车时间是有一定的范围的,它就是[ ti , ti + m),因为当超出这个范围时要么是等车的人还没来,要么等车的人能乘坐上一班车子前往目的地,所以我们可以考虑一个状态:f[i][j] 表示第 i 个人乘车切在 ti + j 的时间中上了车的最小等车时间,其状态必须保证 i + 1 这个人无法上车,且 i 之前可能有人跟 i 一起在车上。
不难发现对于 i 之后的一个位置 k 其中这段区间 [i+1 , k] 的人上车时间可以进行计算,tmp 表示这段人的上车时间 = t[i] + j + m - t[i+k],所以我们出现了最原始的转移:
f[k][tmp] = f[i][j] + Σi+1<=x<=k(tmp + t[k] - t[x])
仍然需要把上方的转移优化成 O(1),使用一个裸的前缀和就可以优化掉了:
f[k][tmp] = f[i][j] + ( tmp + t[k] ) × ( k − i ) − ( sumt[k] − sumt[i] )
最后这个方程仍然可以进行优化,通过把 "我为人人" -> "人人为我" 就可以再一次进行斜率优化时间复杂度就可以变为 O(n),本人比较懒开始拒绝思考了。
代码:
Round1: #include<bits/stdc++.h>
#define MAXT 4000505
using namespace std; int n, m, cnt[MAXT], sum[MAXT], f[MAXT], que[MAXT], Maxt; int main(){
scanf("%d%d", &n, &m);
for (int i=; i<=n; i++){
int x;
scanf("%d", &x);
cnt[x]++, sum[x]+=x;
Maxt=max(Maxt, x);
}
for (int i=; i<Maxt+m; i++) cnt[i]+=cnt[i-], sum[i]+=sum[i-];
int head=, tail=;
for (int i=; i<Maxt+m; i++){
if (i>=m) {
while (head<tail && (f[que[tail]]+sum[que[tail]]-f[que[tail-]]-sum[que[tail-]])*(cnt[i-m]-cnt[que[tail]])>=(f[i-m]+sum[i-m]-f[que[tail]]-sum[que[tail]])*(cnt[que[tail]]-cnt[que[tail-]])) --tail;
que[++tail]=i-m;
}
while (head<tail && i*(cnt[que[head+]]-cnt[que[head]])>=(f[que[head+]]+sum[que[head+]]-f[que[head]]-sum[que[head]])) ++head;
f[i]=cnt[i]*i-sum[i];
if (head<=tail) f[i]=min(f[i], f[que[head]]+(cnt[i]-cnt[que[head]])*i-(sum[i]-sum[que[head]]));
}
int ans=;
for (int i=Maxt; i<Maxt+m; i++) ans=min(ans, f[i]);
printf("%d\n", ans);
return ;
}
Round2: #include<bits/stdc++.h>
using namespace std;
int t[],s[],f[][];
const int inf=;
inline int read()
{
int neg=,x=;
char c;
while((c=getchar())<''||c>'')
if(c=='-')
neg=-;
x=c-'';
while((c=getchar())>=''&&c<='')
x=x*+(c-'');
return x*neg;
}
int main()
{
int n=read(),m=read();
for(int i=;i<=n;i++)
t[i]=read();
sort(t+,t+n+);
for(int i=;i<=n;i++)
s[i]=s[i-]+t[i];
memset(f,0x7f,sizeof(f));
t[]=-inf;
f[][]=;
for(int i=;i<=n;i++)
{
int MAX=min(m-,t[i+]-t[i]);
for(int j=;j<=MAX;j++)
if(f[i][j]!=inf)
for(int k=;i+k<=n;k++)
{
int tmp=max(t[i]+j+m-t[i+k],);
f[i+k][tmp]=min(f[i+k][tmp],f[i][j]+(tmp+t[i+k])*k-(s[i+k]-s[i]));
}
}
int ans=inf;
for(int i=;i<m;i++)
ans=min(ans,f[n][i]);
printf("%d\n",ans);
return ;
}
Luogu 5017 NOIP2018普及组T3 摆渡车 (斜率优化 + 必要的转移进行剪枝)的更多相关文章
- NOIp2018普及组T3暨洛谷P5017 摆渡车:题解
题目链接:https://www.luogu.org/problemnew/show/P5017 emm,这次的真的不简单的,T3比T4难? 醉了... 蒟蒻肯定没有其他大佬讲的好啊,但肯定尽力,真的 ...
- P5017 [NOIP2018 普及组] 摆渡车
P5017 [NOIP2018 普及组] 摆渡车 题目 P5017 思路 将实际问题抽象后,不难发现这是一个 区间 \(DP\) 我们不妨认为时间是一条数轴,每名同学按照到达时刻分别对应数轴上可能重合 ...
- NOIP2018普及组复赛游记
2018年11月10日,NOIP2018普及组复赛. 这是我初中阶段最后一次复赛了. 和往常一样,我们在预定的早上7点,没有出发. 10分钟之后,人终于到齐了,于是出发了,一路无话. 到了南航,合照三 ...
- NOIP2018普及组初赛解题报告
本蒟蒻参加了今年的NOIP2018普及组的初赛 感觉要凉 总而言之,今年的题要说完全没有难度倒也不至于,还有不少拼RP的题,比如第一次问题求解考逻辑推理,第一次完善程序考双链表等 下面我就和大家一起看 ...
- P5018 [NOIP2018 普及组] 对称二叉树
P5018 [NOIP2018 普及组] 对称二叉树 题目 P5018 思路 通过hash值来判断左右树是否相等 \(hl[i]\) 与 \(Hl[i]\) 是防止hash冲突, \(r\) 同理 注 ...
- NOIP2012 普及组 T3 摆花——S.B.S.
题目描述 小明的花店新开张,为了吸引顾客,他想在花店的门口摆上一排花,共m盆.通过调查顾客的喜好,小明列出了顾客最喜欢的n种花,从1到n标号.为了在门口展出更多种花,规定第i种花不能超过ai盆,摆花时 ...
- NOIP2008 普及组T3 传球游戏 解题报告-S.B.S.
题目描述 上体育课的时候,小蛮的老师经常带着同学们一起做游戏.这次,老师带着同学们一起做传球游戏. 游戏规则是这样的:n个同学站成一个圆圈,其中的一个同学手里拿着一个球,当老师吹哨子时开始传球,每个同 ...
- NOIP2011 普及组 T3 洛谷P1309 瑞士轮
今天题做太少,放道小题凑数233 题目背景 在双人对决的竞技性比赛,如乒乓球.羽毛球.国际象棋中,最常见的赛制是淘汰赛和循环赛.前者的特点是比赛场数少,每场都紧张刺激,但偶然性较高.后者的特点是较为公 ...
- NOIP2018普及组模拟赛
向老师给的模拟赛,还没普及组难... 题目在洛谷团队里. 第一试三道水题,我46分钟就打完了,然后就AK了. 第二试一看,除了第二题要思考一段时间之外,还是比较水的,但是我得了Rank倒1,115分. ...
随机推荐
- Codeforces 1137D(技巧)
一开始写的第一步让0和1一起走然后第二步再让0走会挂最后一个点--然后探索一下觉得主要问题在于我模拟的一步一步地走.如果这样的话9 2这个数据会使第17步他俩就碰在final点了,而实际上我们想要的效 ...
- 洛谷P1896||bzoj1087 [SCOI2005]互不侵犯
bzoj1087 洛谷P1896 想了很久,太久没做状压都已经不会了... 状压每一行就好了 #include<cstdio> #include<algorithm> #inc ...
- SQL Server 查看列,添加列,修改列,删除列
查看表:exec sp_help 表名 查看列: exec sp_columns 表名 查看列:select * from information_schema.columns where table ...
- android开发学习 ------- 仿QQ侧滑效果的实现
需要做一个仿QQ侧滑删除的一个效果: 一开始是毫无头绪,百度找思路,找到 https://blog.csdn.net/xiaxiazaizai01/article/details/53036994 ...
- 海德汉S3078长度计
手头有一款海德汉S3078的增量式长度计. 图三为长度计引出线缆的针脚编号,其中,Up和传感器Up短路,0V与传感器0V短路,其他为输出信号线缆.时序如图: Ua1和Ua2两个信号的相位,可以得出长度 ...
- ABC时间管理法
名称 ABC时间管理法 属于 事务优先顺序法的“鼻祖” 做法 将待办的事项按照又重要到轻的顺序划分为A,B,C三个等级,然后按照事项的重要等级依据完成任务的做事方法. 特点 使学习.工作和生活等活动在 ...
- AJAX的JSON方式传回方法
AJAX返回数据的类型有两种,一种是TEXT类型,一种是JSON类型. 使用TEXT类型,访问数据库后将结果拼接成字符串,返回时在拆分成数组使用. JSON直接将结果转成JSON数据,返回时直接使用. ...
- IE下的圆角
元素{ position: relative;/*必须*/ z-index: 10;/*必须*/ border-radius: 8px; -moz-border-radius: 8px; -webki ...
- Scanner-String-StringBuilder-API
1.能够明确API的使用步骤 1)打开帮助文档 2)点击显示,找到索引,看到输入框 3)你要找谁?在输入框里输入,然后回车 4)看包:java.lang下的类不需 ...
- SPM-软件项目管理之感想
这学期开始选择选修课的时候是需要把每节课都过一遍的.当我上完SPM那节课的时候,我就已经决定要选这门课了,尽管还有其他的课我都还没上过.由于这节课是双语教学-中文老师兼外籍老师,这样的方式感觉很新颖, ...