POJ 1661 暴力dp
题意略。
思路:
很有意思的一个题,我采用的是主动更新未知点的方式,也即刷表法来dp。
我们可以把整个路径划分成横向移动和纵向移动,题目一开始就给出了Jimmy的高度,这就是纵向移动的距离。
我们dp的目标是每个线段端点的横向移动最小值。
有一个小trick,就是有可能Jimmy开始就可以落在地上,没有必要或者说在下落过程中不会有间隔来阻拦。
代码附上:
#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
using namespace std;
const int maxn = ;
const int INF = 0x3f3f3f3f;
const int F = 0x3f; struct segment{
int l,r,h;
segment(int l = ,int r = ,int h = ){
this->l = l,this->r = r,this->h = h;
}
}; segment store[maxn];
int dp[maxn<<],N,x,h,MAX,t;
bool covered[maxn<<]; bool cmp(const segment& s0,const segment& s1){
return s0.h > s1.h;
}
bool inSegment(int x,segment s){
return s.l <= x && x <= s.r;
} int main(){
scanf("%d",&t);
while(t--){
scanf("%d%d%d%d",&N,&x,&h,&MAX);
for(int i = ;i < N;++i) scanf("%d%d%d",&store[i].l,&store[i].r,&store[i].h);
memset(dp,F,sizeof(dp));
memset(covered,false,sizeof(covered));
sort(store,store + N,cmp);
int ans = h,lft,rht,lp,rp,nlft,nrht,nlp,nrp;
for(int i = ;i < N;++i){
lft = i<<,rht = lft + ;
if(inSegment(x,store[i])){
dp[lft] = x - store[i].l,dp[rht] = store[i].r - x;
break;
}
}
for(int i = ;i < N;++i){
lft = i<<,rht = lft + ;
lp = store[i].l,rp = store[i].r;
int cnt0 = ,cnt1 = ;
if(dp[lft] == INF) continue;
for(int j = i + ;store[i].h - store[j].h <= MAX && j < N && cnt0 + cnt1 < ;++j){
if(store[i].h == store[j].h) continue;
nlft = j<<,nrht = nlft + ;
nlp = store[j].l,nrp = store[j].r;
if(!cnt0 && inSegment(lp,store[j])){
dp[nlft] = min(dp[nlft],dp[lft] + lp - nlp);
dp[nrht] = min(dp[nrht],dp[lft] + nrp - lp);
cnt0 = ;
}
if(!cnt1 && inSegment(rp,store[j])){
dp[nlft] = min(dp[nlft],dp[rht] + rp - nlp);
dp[nrht] = min(dp[nrht],dp[rht] + nrp - rp);
cnt1 = ;
}
}
} for(int i = ;i < N;++i){
if(store[i].h > MAX) continue;
for(int j = i + ;j < N;++j){
if(inSegment(store[i].l,store[j])) covered[i<<] = true;
if(inSegment(store[i].r,store[j])) covered[i<< | ] = true;
}
} int minn = INF;
for(int i = N - ;i >= && store[i].h <= MAX;--i){
lft = i<<,rht = lft + ;
if(!covered[lft]) minn = min(minn,dp[lft]);
if(!covered[rht]) minn = min(minn,dp[rht]);
}
if(minn == INF) minn = ;
ans += minn;
printf("%d\n",ans);
}
return ;
} /* 1
3 4 4 5
3 5 3
1 7 2
2 6 1
*/
POJ 1661 暴力dp的更多相关文章
- POJ 1661 Help Jimmy(递推DP)
思路: 1. 每个板子有左右两端, dp[i][0], dp[i][1] 分别记录左右端到地面的时间 2. 从下到上递推计算, 上一层的板子必然会落到下面的某一层板子上, 或者地面上 总结: 1. 计 ...
- E - Help Jimmy POJ - 1661 dp
E - Help Jimmy POJ - 1661 这个题目本身不是很难,但是可以更加优化这个写法. 开始是n*n #include <cstdio> #include <cstri ...
- POJ 2182/暴力/BIT/线段树
POJ 2182 暴力 /* 题意: 一个带有权值[1,n]的序列,给出每个数的前面比该数小的数的个数,当然比一个数前面比第一个数小的个数是0,省略不写,求真正的序列.(拗口) 首先想到的是从前到后暴 ...
- Luogu P1436 棋盘分割 暴力DP
我的天,,,,,n=8,k<=15,,,这怕不是暴力DP+高维数组.... 开一个五维数组f[k][i][j][p][q]表示从(i,j)到(p,q)中分成k个矩形最小的平方和. 然后初始化时用 ...
- Fire (poj 2152 树形dp)
Fire (poj 2152 树形dp) 给定一棵n个结点的树(1<n<=1000).现在要选择某些点,使得整棵树都被覆盖到.当选择第i个点的时候,可以覆盖和它距离在d[i]之内的结点,同 ...
- 暴力/DP Codeforces Beta Round #22 (Div. 2 Only) B. Bargaining Table
题目传送门 /* 题意:求最大矩形(全0)的面积 暴力/dp:每对一个0查看它左下的最大矩形面积,更新ans 注意:是字符串,没用空格,好事多磨,WA了多少次才发现:( 详细解释:http://www ...
- 【2019.8.8 慈溪模拟赛 T1】开箱(chest)(暴力DP水过)
转化题意 这题目乍一看十分玄学,完全不可做. 但实际上,假设我们在原序列从小到大排序之后,选择开的宝箱编号是\(p_{1\sim Z}\),则最终答案就是: \[\sum_{i=1}^Za_{p_i} ...
- 【2019.8.12 慈溪模拟赛 T1】钥匙(key)(暴力DP)
暴力\(DP\) 这题做法很多,有\(O(n^2)\)的,有\(O(n^2logn)\)的,还有徐教练的\(O(nlogn)\)的,甚至还有\(bzt\)的二分+线段树优化建图的费用流. 我懒了点,反 ...
- POJ 1661 Help Jimmy(C)动态规划
没刷过 POJ,这题是论坛有人问的,我才看看. 我发现 POJ 注册很奇怪,账号总是登不上去,弄的我还注册两个.Emmm 首次体验很差,还好我不在 POJ 刷题. 题目链接:POJ 1661 Help ...
随机推荐
- Vue的基本使用(四)
1.refs属性的使用 <!DOCTYPE html> <html lang="zh-CN"> <head> <meta charset= ...
- eclipse(java windows)
百度云:链接:http://pan.baidu.com/s/1i4Zjv97 密码:u0qh 官方下载网址:http://www.eclipse.org/downloads/eclipse-pa ...
- linux初学者-普通磁盘分区篇
linux初学者-普通磁盘分区篇 磁盘是计算机的重要组成部分,是记录数据的场所.在使用磁盘时,经常需要对其进行分区来实现不同的用途.下文将介绍在linux系统中普通磁盘分区的方法. "fdi ...
- Go语言圣经习题练习_1.5. 获取URL
练习 1.7: 函数调用io.Copy(dst, src)会从src中读取内容,并将读到的结果写入到dst中,使用这个函数替代掉例子中的ioutil.ReadAll来拷贝响应结构体到os.Stdout ...
- JavaOOP 对象和封装
1.后缀:jsp---相当于html,但是它里面可以写java代码. 2.包名取名规则 a.网站域名倒着写 b.字母小写 3.类名取名规则 a.首字母大写 4.三目运算(适用简单的if-else) 条 ...
- 努力做一个优秀的programmer [ C# 影院售票系统]
Cinema.cs类 [Serializable] // 电影院类 public class Cinema { public Cinema() { //二进制 SoldTickets = new Li ...
- 基于redis的消息订阅与发布
Redis 的 SUBSCRIBE 命令可以让客户端订阅任意数量的频道, 每当有新信息发送到被订阅的频道时, 信息就会被发送给所有订阅指定频道的客户端. 作为例子, 下图展示了频道 channel1 ...
- 又一个轮子--QMapper
1 前言 我喜欢造轮子,一是造的时候就是深刻学习的时候,二是造着造着,说不定某天比世面上的其它轮子都要好呢.比如造过Networksocket,也造过WebApiClient,现在我也要造一个Mapp ...
- c#小灶——初识c#
提到c#,就不得不说.net,.net是微软开发的一个平台,简单来说,在这个平台上,可以编写.运行程序.可能很多人觉得这个平台离我们很遥远,其实不然,这个平台就一直在我们的windows操作系统里,默 ...
- Spring的依赖注入和管理Bean
采用Spring管理Bean和依赖注入 1.实例化spring容器 和 从容器获取Bean对象 实例化Spring容器常用的两种方式: 方法一: 在类路径下寻找配置文件来实例化容器 [推荐使用] Ap ...