[UVALive 3983] Robotruck
图片加载可能有点慢,请跳过题面先看题解,谢谢



设状态 \(f[i][j]\) 为,当前垃圾序号为 \(i\) ,当前承重为 \(j\) 的最小路程,好的这道题做完了
O(NC) G烂
$
$
我们这样设: \(f[i]\) 为,从起点出发,将前 \(i\) 个垃圾清完并放进垃圾桶的最小路程
显然有转移:\(f[i]=min(f[j]+H[j+1]+dis[j+1][i]+H[i]),j\le i,sum[j+1][i]<=c\),其中,
\(sum[i][j]\) 为第 \(i\) 个垃圾到第 \(j\) 个垃圾的重量和,\(H[i]\) 为垃圾 \(i\) 到原点的距离,\(dis[i][j]\) 为从第 \(i\) 个垃圾出发,依次经过第 \(i+1,i+2...j\) 个垃圾的总路程
我们将上面的式子转换一下,设 \(t[i]\) 为,从第 \(1\) 个垃圾出发,依次经过第 \(2,3,4...i\) 个垃圾的总路程,
则有:\(dis[i][j]=t[j]-t[i]\)
所以:\(f[i]=min(f[j]-t[j+1]+H[j+1])+t[i]+H[i],sum[j+1][i]<=c\)
令:\(calc(j)=f[j]-t[j+1]+H[j+1]\),所以有:\(f[i]=min(calc(j))+t[i]+H[i],sum[j+1][i]<=c\)
这个式子。。。用什么数据结构都能维护。。。\(XJB\) 搞一下就能 \(AC\) 了
$
$
这里用到单调队列,时间复杂度:\(O(n)\)
//made by Hero_of_Someone
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cmath>
#define N (100010)
#define il inline
#define RG register
using namespace std;
il int gi(){ RG int x=0,q=1; RG char ch=getchar(); while( ( ch<'0' || ch>'9' ) && ch!='-' ) ch=getchar();
if( ch=='-' ) q=-1,ch=getchar(); while(ch>='0' && ch<='9') x=x*10+ch-48,ch=getchar(); return q*x; }
int T,n,c;
int x[N],y[N];
int t[N],S[N],H[N];
int que[N],f[N];
il void init(){
c=gi(),n=gi();
for(RG int i=1;i<=n;i++){
x[i]=gi(),y[i]=gi(); int w=gi();
H[i]=abs(x[i])+abs(y[i]);
t[i]=t[i-1]+abs(x[i]-x[i-1])+abs(y[i]-y[i-1]);
S[i]=S[i-1]+w;
}
}
il int calc(int x){return f[x]-t[x+1]+H[x+1];}
il void work(){
int hd=0,tl=0;
for(RG int i=1;i<=n;i++){
while(hd<=tl && S[i]-S[que[hd]]>c) hd++;
f[i]=calc(que[hd])+t[i]+H[i];
while(hd<=tl && calc(i)<=calc(que[tl])) tl--;
que[++tl]=i;
}
printf("%d\n",f[n]); if(T)puts("");
}
int main(){ T=gi(); while(T--){ init(); work(); } return 0; }
$
$
附,优先队列的代码,时间复杂度:\(O(nlogn)\)
//made by Hero_of_Someone
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<queue>
#include<cmath>
#define N (100010)
#define il inline
#define RG register
using namespace std;
il int gi(){ RG int x=0,q=1; RG char ch=getchar(); while( ( ch<'0' || ch>'9' ) && ch!='-' ) ch=getchar();
if( ch=='-' ) q=-1,ch=getchar(); while(ch>='0' && ch<='9') x=x*10+ch-48,ch=getchar(); return q*x; }
int T,n,c;
int x[N],y[N];
int t[N],S[N],H[N];
int f[N];
il void init(){
c=gi(),n=gi();
for(RG int i=1;i<=n;i++){
x[i]=gi(),y[i]=gi(); int w=gi();
H[i]=abs(x[i])+abs(y[i]);
t[i]=t[i-1]+abs(x[i]-x[i-1])+abs(y[i]-y[i-1]);
S[i]=S[i-1]+w;
}
}
struct R{int id,x;
il bool operator<(const R &a)const{return x>a.x;}
}a;
priority_queue<R>que;
il void work(){
while(!que.empty()) que.pop();
que.push(a);
for(RG int i=1;i<=n;i++){
R x=que.top();
while(!que.empty()&&S[i]-S[x.id]>c) que.pop(),x=que.top();
f[i]=x.x+t[i]+H[i];
que.push((R){i,f[i]-t[i+1]+H[i+1]});
}
printf("%d\n",f[n]); if(T)puts("");
}
int main(){ T=gi(); while(T--){ init(); work(); } return 0; }
$
$
另附,在vjudge上跑的比STL还慢的手写堆代码:
//made by Hero_of_Someone
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<queue>
#include<cmath>
#define N (100010)
#define il inline
#define RG register
using namespace std;
il int gi(){ RG int x=0,q=1; RG char ch=getchar(); while( ( ch<'0' || ch>'9' ) && ch!='-' ) ch=getchar();
if( ch=='-' ) q=-1,ch=getchar(); while(ch>='0' && ch<='9') x=x*10+ch-48,ch=getchar(); return q*x; }
int T,n,c;
int x[N],y[N];
int t[N],S[N],H[N];
int f[N],dis[N];
struct heap{
#define fa (x>>1)
#define ls (x<<1)
#define rs (x<<1|1)
int a[N],id[N],len;
il int top(){ return a[1]; }
il void push(RG int u){
if(!id[u]) id[u]=++len,a[len]=u; RG int x=id[u];
while(fa){
if(dis[a[x]]>=dis[a[fa]]) break;
swap(a[x],a[fa]),id[a[x]]=x,id[a[fa]]=fa,x=fa;
}
return;
}
il void pop(){
id[a[1]]=0,a[1]=a[len--]; if (len) id[a[1]]=1; RG int x=1,son;
while(ls<=len){
son=(rs<=len && dis[a[rs]]<dis[a[ls]]) ? rs : ls;
if(dis[a[x]]<=dis[a[son]]) break;
swap(a[x],a[son]),id[a[x]]=x,id[a[son]]=son,x=son;
}
return;
}
#undef fa
#undef ls
#undef rs
}que;
il void init(){
c=gi(),n=gi();
for(RG int i=1;i<=n;i++){
x[i]=gi(),y[i]=gi(); int w=gi();
H[i]=abs(x[i])+abs(y[i]);
t[i]=t[i-1]+abs(x[i]-x[i-1])+abs(y[i]-y[i-1]);
S[i]=S[i-1]+w;
}
}
il void work(){
while(que.len) que.pop();
que.push(0);
for(RG int i=1;i<=n;i++){
int x=que.top();
while(que.len&&S[i]-S[x]>c) que.pop(),x=que.top();
f[i]=dis[x]+t[i]+H[i];
dis[i]=f[i]-t[i+1]+H[i+1]; que.push(i);
}
printf("%d\n",f[n]); if(T)puts("");
}
int main(){ T=gi(); while(T--){ init(); work(); } return 0; }
[UVALive 3983] Robotruck的更多相关文章
- 【暑假】[深入动态规划]UVAlive 3983 Robotruck
UVAlive 3983 Robotruck 题目: Robotruck Time Limit: 3000MS Memory Limit: Unknown 64bit IO Format ...
- UVALive 3983 Robotruck (单调队列,dp)
如果状态定义为序号和重量的话,决策就是下一个垃圾捡或者不减,但是状态数太多了. 如果只定义序号作为状态的话,决策就变成从前面的某个j一直捡到i才送回垃圾. 这就变成了一个区间选最小值的问题,用单调队列 ...
- UVaLive 3983 Robotruck (DP + 单调队列)
题意:有n个垃圾,第i个垃圾坐标为(xi,yi),重量为wi,有一个机器人,要按照编号从小到大的顺序剑气所有的垃圾兵扔进垃圾桶,垃圾桶在原点, 每次总重量不能超过C,两点间距离为曼哈顿距离,求出最短的 ...
- LA 3983 Robotruck
这道题感觉挺吃力的,还用到了我不熟悉的优先队列 题目中的推导也都看明白了,总之以后还要多体会才是 这里用优先对列的原因就是因为要维护一个滑动区间的最小值,比如在区间里2在1的前面,2在离开这个滑动区间 ...
- UVA LA 3983 - Robotruck DP,优先队列 难度: 2
题目 https://icpcarchive.ecs.baylor.edu/index.php?option=com_onlinejudge&Itemid=8&page=show_pr ...
- UVALive 3983 捡垃圾的机器人 DP
这个题目我最初的做法沿用树形DP的做法,设置一个 dp[i][0]表示机器人在i点不回去的最短路径,dp[i][1]表示机器人在i点回去的最短路径,规划方向为i-1向i转移,结果发现这个不能用树形的结 ...
- 大白第一章第四节dp例题
入口 UVALive - 3882 #include<cstdio> using namespace std; ; int n,m,k,f[N]; int main(){ //f[i]表示 ...
- UVALive - 4108 SKYLINE[线段树]
UVALive - 4108 SKYLINE Time Limit: 3000MS 64bit IO Format: %lld & %llu Submit Status uDebug ...
- UVALive - 3942 Remember the Word[树状数组]
UVALive - 3942 Remember the Word A potentiometer, or potmeter for short, is an electronic device wit ...
随机推荐
- bootstrap switch样式修改与多列等间距布局
先以一张图开启今天的随笔 今天实习遇到了switch按钮,小姐姐说用插件bootstrap switch来写,我第一次用这个插件,首先在引入方面就遇到了很多坑,先来总结一下bootstrap swit ...
- 你也可以手绘二维码(二)纠错码字算法:数论基础及伽罗瓦域GF(2^8)
摘要:本文讲解二维码纠错码字生成使用到的数学数论基础知识,伽罗瓦域(Galois Field)GF(2^8),这是手绘二维码填格子理论基础,不想深究可以直接跳过.同时数论基础也是 Hash 算法,RS ...
- 三羊献瑞:dfs / next_permutation()
三羊献瑞 观察下面的加法算式: 祥 瑞 生 辉 + 三 羊 献 瑞------------------- 三 羊 生 瑞 气 (如果有对齐问题,可以参看[图1.jpg]) 其中,相同的汉字代 ...
- [译文]c#扩展方法(Extension Method In C#)
原文链接: https://www.codeproject.com/Tips/709310/Extension-Method-In-Csharp 介绍 扩展方法是C# 3.0引入的新特性.扩展方法使你 ...
- $.each()用法
通过它,你可以遍历对象.数组的属性值并进行处理. 使用说明 each函数根据参数的类型实现的效果不完全一致: 1.遍历对象(有附加参数) $.each(Object, function(p1, p2) ...
- Beta发布文案+美工
团队名称:探路者 1蔺依铭:http://www.cnblogs.com/linym762/(组长) 2张恩聚:http://www.cnblogs.com/zej87/ 3米赫:http://www ...
- 团队博客作业Week4 --- 学霸网站--NABC
1.需求(Need) 伴随着经济的发展,科学技术取得了飞速的发展,互联网在各行各业的发展中取得了广泛的应用.随着这些事物的发展,我们每个人都会接触到相当庞大的数据.如何在这些数据中找到自己需要的,如何 ...
- 3、昨天的BUG
基本功能实现了,但是有一些小问题,修改昨天余留的BUG
- 安卓开发神器vysor+adb wifi
准备: 1.vysor需要FQ从google应用商店下载,装在google上,目前知道的免费的vysor的作用是电脑显示手机屏幕并且能操控手机. 步骤:FQ后就能下载了,FQ方法不赘述.
- IIs8 svc
IIS8中添加WCF支持几种方法小结[图文] 方法一 最近在做Silverlight,Windows Phone应用移植到Windows 8平台,在IIS8中测试一些传统WCF服务应用,发现IIS8不 ...