Loj2604开车旅行

我完全没有看出这道题哪里是DP

首先,一个位置向后的最近和第二近我们可以通过set去简单实现

通过维护最大和次大即可

至于高度相同的情况我们可以通过先在set中查询小的来实现

接下来我们考虑倍增

\(f_{i,j}\)表示从位置\(j\)开始向后开\(2^i\)次所到达的位置(这里一次的定义是小A走一次然后小B再走一次)

我们设\(g1_{i,j}\)表示从\(j\)开始向后走\(2^i\)步的过程中小A走的路程,\(g2_{i,j}\)表示小B的

我们每次对于一个\(s_i,x_i\)我们尝试在\(s_i\)位置倍增向后跳,跳到马上大于\(x_i\)或者无路可走位置

如果无路可走,说明剩下的路程不满足走一次,但是可能出现小A还能走得情况,所以要特判

#include<cstdio>
#include<iostream>
#include<cstring>
#include<cctype>
#include<set>
#include<queue>
#include<cmath>
#include<algorithm>
#define pii pair<int,int>
#define mk make_pair
#define fi first
#define se second
#define LL long long
using namespace std;
const int N = 2e5 + 3;
const int INF = 2e9;
int n,m;
int h[N];
pii nxt1[N],nxt2[N];
int f[21][N];
LL g1[21][N],g2[21][N];
set < pii > s;
set < pii >::iterator it;
inline int read(){
int v = 0,c = 1;char ch = getchar();
while(!isdigit(ch)){
if(ch == '-') c = -1;
ch = getchar();
}
while(isdigit(ch)){
v = v * 10 + ch - 48;
ch = getchar();
}
return v * c;
}
int main(){
n = read();
for(int i = 1;i <= n;++i) h[i] = read();
for(int i = n;i >= 1;--i){
pii f1 = mk(INF,0),f2 = mk(INF,0);LL v = 0;
it = s.lower_bound(mk(h[i],i));
if(it != s.begin()){
it--;
v = abs((*it).fi - h[i]);
if(v < f1.fi) f2 = f1,f1 = mk(v,(*it).se);
else if(v < f2.fi) f2 = mk(v,(*it).se);
if(it != s.begin()){
it--;
v = abs((*it).fi - h[i]);
if(v < f1.fi) f2 = f1,f1 = mk(v,(*it).se);
else if(v < f2.fi) f2 = mk(v,(*it).se);
}
}
it = s.upper_bound(mk(h[i],i));
if(it != s.end()){
v = abs((*it).fi - h[i]);
if(v < f1.fi) f2 = f1,f1 = mk(v,(*it).se);
else if(v < f2.fi) f2 = mk(v,(*it).se);
it++;
if(it != s.end()){
v = abs((*it).fi - h[i]);
if(abs((*it).fi - h[i]) < f1.fi) f2 = f1,f1 = mk(v,(*it).se);
else if(v < f2.fi) f2 = mk(v,(*it).se);
}
}
if(f1.se) nxt1[i] = f1;
if(f2.se) nxt2[i] = f2;
s.insert(mk(h[i],i));
}
for(int i = 1;i <= n;++i){
// printf("1s:%d 1id:%d 2s:%d 2id:%d\n",nxt1[i].fi,nxt1[i].se,nxt2[i].fi,nxt2[i].se);
f[0][i] = nxt1[nxt2[i].se].se;
g1[0][i] = nxt2[i].fi;
g2[0][i] = nxt1[nxt2[i].se].fi;
// printf("%d %lld %lld\n",f[0][i],g1[0][i],g2[0][i]);
}
for(int i = 1;i < 20;++i){
for(int j = 1;j <= n;++j){
f[i][j] = f[i - 1][f[i - 1][j]];
if(f[i][j]){
g1[i][j] = g1[i - 1][j] + g1[i - 1][f[i - 1][j]];
g2[i][j] = g2[i - 1][j] + g2[i - 1][f[i - 1][j]];
}
}
}
// printf("%d %lld %lld\n",f[1][1],g1[1][1],g2[1][1]);
LL x0 = read();int maxxhi = 0,id = 0;LL zi = 0,mu = 0;
for(int i = 1;i <= n;++i){
LL z1 = 0,m1 = 0,tt = 0;int x = i;
for(int j = 19;j >= 0;--j){
if(f[j][x]){
if(tt + g1[j][x] + g2[j][x] <= x0){
tt += g1[j][x] + g2[j][x];
z1 += g1[j][x],m1 += g2[j][x],x = f[j][x];
}
}
}
if(nxt2[x].se && tt + nxt2[x].fi <= x0) tt += nxt2[x].fi,z1 += nxt2[x].fi;
if(m1 == 0){
if(mu != 0) continue;
if(maxxhi < h[i]) zi = z1,mu = m1,maxxhi = h[i],id = i;
}
else{
if(mu == 0){
zi = z1,mu = m1,maxxhi = h[i],id = i;
continue;
}
if(z1 * mu < zi * m1) zi = z1,mu = m1,maxxhi = h[i],id = i;
else if(z1 * mu == zi * m1 && maxxhi < h[i]) zi = z1,mu = m1,maxxhi = h[i],id = i;
}
}
printf("%d\n",id);
m = read();
while(m--){
int si = read(),xi = read();
LL s1 = 0,s2 = 0;int x = si;LL tt = 0;
for(int j = 19;j >= 0;--j){
if(f[j][x]){
if(tt + g1[j][x] + g2[j][x] <= xi){
tt += g1[j][x] + g2[j][x];
s1 += g1[j][x],s2 += g2[j][x],x = f[j][x];
}
}
}
if(nxt2[x].se && tt + nxt2[x].fi <= xi) s1 += nxt2[x].fi;
printf("%lld %lld\n",s1,s2);
}
return 0;
}

Loj2604开车旅行的更多相关文章

  1. Luogu 1081 【NOIP2012】开车旅行 (链表,倍增)

    Luogu 1081 [NOIP2012]开车旅行 (链表,倍增) Description 小A 和小B决定利用假期外出旅行,他们将想去的城市从1到N 编号,且编号较小的城市在编号较大的城市的西边,已 ...

  2. CH5701 开车旅行

    题意 5701 开车旅行 0x50「动态规划」例题 描述 小A和小B决定利用假期外出旅行,他们将想去的城市从1到N编号,且编号较小的城市在编号较大的城市的西边,已知各个城市的海拔高度互不相同,记城市 ...

  3. 2012Noip提高组Day1 T3 开车旅行

    题目描述 小 A 和小 B 决定利用假期外出旅行,他们将想去的城市从 1 到 N 编号,且编号较小的 城市在编号较大的城市的西边,已知各个城市的海拔高度互不相同,记城市 i 的海拔高度为 Hi,城市 ...

  4. Cogs 1264. [NOIP2012] 开车旅行(70分 暴力)

    1264. [NOIP2012] 开车旅行 ★★☆   输入文件:drive.in   输出文件:drive.out   简单对比时间限制:2 s   内存限制:128 MB [题目描述] 小A 和小 ...

  5. 开车旅行 【NOIP2012 D1T3】

    开车旅行 [NOIP2012 D1T3] 倍增 首先令\(a[i]\)表示从i出发最近的城市下标,\(b[i]\)表示从i出发第二近的城市下标 可以维护一个\(\text{set<pair< ...

  6. 洛谷 P1081 开车旅行(70)

    P1081 开车旅行 题目描述 小AA 和小BB 决定利用假期外出旅行,他们将想去的城市从 11到 NN 编号,且编号较小的城市在编号较大的城市的西边,已知各个城市的海拔高度互不相同,记城市 ii的海 ...

  7. 开车旅行 2012年NOIP全国联赛提高组(倍增+set)

    开车旅行 2012年NOIP全国联赛提高组  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond     题目描述 Description 小A 和小B决定利用 ...

  8. [NOIP2012提高组]开车旅行

    题目:洛谷P1081.Vijos P1780.codevs1199. 题目大意:有n座海拔高度不相同的城市(编号1~n),两城市的距离就是两城市海拔之差.规定每次只能从编号小的城市走到编号大的城市. ...

  9. P1081 [NOIP2012]开车旅行[倍增]

    P1081 开车旅行    题面较为啰嗦.大概概括:一个数列,只能从一个点向后走,两种方案:A.走到和自己差的绝对值次小的点B.走到和自己差的绝对值最小点:花费为此差绝对值:若干询问从规定点向后最多花 ...

随机推荐

  1. Directx11教程(66) D3D11屏幕文本输出(1)

    原文:Directx11教程(66) D3D11屏幕文本输出(1)      在D3D10中,通过ID3DX10Font接口对象,我们可以方便的在屏幕上输出文字信息,一个DrawText函数就能解决所 ...

  2. 支付宝sdk iOS 集成

    1添加支付宝源文件和库文件AlipayOrder.h    AlipayOrder.m    AlipayResult.h    AlipayResult.m  AlixLibService.h   ...

  3. 2019-10-26-Inno-Setup-安装包脚本-Run-的-Flags-标记

    title author date CreateTime categories Inno Setup 安装包脚本 Run 的 Flags 标记 lindexi 2019-10-26 08:42:39 ...

  4. 比较全面的一个PHP缓存类解析

    转自:http://www.blhere.com/1164.html 一.引论 PHP,一门最近几年兴起的web设计脚本语言,由于它的强大和可伸缩性,近几年来得到长足的发展,php相比传统的asp网站 ...

  5. 【JZOJ4883】【NOIP2016提高A组集训第12场11.10】灵知的太阳信仰

    题目描述 在炽热的核熔炉中,居住着一位少女,名为灵乌路空. 据说,从来没有人敢踏入过那个熔炉,因为人们畏缩于空所持有的力量--核能. 核焰,可融真金. 咳咳. 每次核融的时候,空都会选取一些原子,排成 ...

  6. 【LINUX】降级安装低版本GCC,G++

    由于要制作crosstool,需要用到gcc 4.1.2来编译,而Ubuntu 12.04下的gcc版本是gcc 4.6.3,高版本的gcc也不是好事啊. 下面介绍gcc 4.1.2的编译安装方法: ...

  7. Java面向对象----封装概念

    封装 信息隐藏,隐藏对象的细节 访问修饰符 public private protected 默认 属性封装的实现 方法封装的目的(隐藏方法实现细节) package com.tanlei.newer ...

  8. UVa 10323 【数学】

    UVa 10323 题目:计算阶乘在10000~6227020800之间的值,不在范围对应输出Under或者Over. 分析:简单题.数论.因为13!=6227020800,7!<10000&l ...

  9. HDU-1114_Piggy-Bank

    Piggy-Bank Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Problem ...

  10. Java练习 SDUT-1294_选票统计

    选票统计 Time Limit: 1000 ms Memory Limit: 65536 KiB Problem Description 某校学生会主席由全校学生投票选举产生,共有m名候选人报名参选, ...