n<=100000个山,每个山有高度,从一个山到另一个山代价为高度差,有A和B两人一起开车,A每次选前进方向的次近山,B选最近,保证山高度不同且如果代价相同的山低的代价算小,每次旅行先A走,然后B,然后AB轮流开车,旅行如果下一次找不到目的地或者下一次到目的地时总代价超过了指定的X,他们就会停下。现完成两个任务:一,告诉X0,问从哪个点开始完成一次预算代价为X0的旅行会使A的路程比B的路程最小;二,m个询问,每次问从Si做预算Xi的旅行,A和B的行驶路程。

首先需要知道每个人后面的最近和次近山,双向链表、set、乱七八糟,随便搞。这里选双向链表,前面需要一个离散化:把每个数的排名连接起来,从1到n扫一次,每次用一个数的前驱、前驱的前驱、后继、后继的后继来比较出最近和次近,然后在双向链表中删除之。

由于A和B走一步的方式不一样,这样“跳”会比较复杂,不如把“跳”一步定义成A和B都走一次,这样可以处理一个ST表,表示从点i跳2^j步跳到哪里,以及期间A和B的行驶路程。用这个可以轻松算出所有的询问。

询问一:枚举从每个点开始跳,看最远能跳到哪,然后最后一步看A能不能再开一次。

询问二:同理。

倍增写挂了,开了个临时变量now来跳结果用i来跳。。。浪费了一晚上

 #include<stdio.h>
#include<string.h>
#include<math.h>
#include<stdlib.h>
#include<assert.h>
#include<algorithm>
//#include<queue>
//#include<iostream>
using namespace std; bool isdigit(char c) {return c>='' && c<='';}
int qread()
{
char c;int s=,t=;while (!isdigit(c=getchar())) (c=='-' && (t=-));
do s=s*+c-''; while (isdigit(c=getchar()));return s*t;
} int n;
#define maxn 100011
int pre[maxn],suc[maxn],zui[maxn],zuilu[maxn],ci[maxn],cilu[maxn],Rank[maxn],lisan[maxn],id[maxn];
#define LL long long
LL hei[maxn],f[maxn][],fa[maxn][],fb[maxn][];
void prepare()
{
for (int i=;i<=n+;i++) pre[i]=i-;
for (int i=;i<=n;i++) suc[i]=i+;
for (int i=;i<=n;i++) lisan[i]=hei[i];
sort(lisan+,lisan++n);
for (int i=;i<=n;i++) Rank[i]=lower_bound(lisan+,lisan++n,hei[i])-lisan,id[Rank[i]]=i;
for (int i=;i<=n;i++) assert(lisan[i]!=lisan[i-]);
// for (int i=1;i<=n;i++) cout<<Rank[i]<<' ';cout<<endl;
hei[]=-1e15;id[]=id[n+]=;
for (int i=;i<=n;i++)
{
LL x,y;
if (fabs((x=hei[id[suc[Rank[i]]]])-hei[i])<fabs((y=hei[id[pre[Rank[i]]]])-hei[i]))
{
zui[i]=id[suc[Rank[i]]];
zuilu[i]=fabs(x-hei[i]);
if (fabs((x=hei[id[suc[suc[Rank[i]]]]])-hei[i])<fabs(y-hei[i]))
{
ci[i]=id[suc[suc[Rank[i]]]];
cilu[i]=fabs(x-hei[i]);
}
else
{
ci[i]=id[pre[Rank[i]]];
cilu[i]=fabs(y-hei[i]);
}
}
else
{
zui[i]=id[pre[Rank[i]]];
zuilu[i]=fabs(y-hei[i]);
if (fabs(x-hei[i])<fabs((y=hei[id[pre[pre[Rank[i]]]]])-hei[i]))
{
ci[i]=id[suc[Rank[i]]];
cilu[i]=fabs(x-hei[i]);
}
else
{
ci[i]=id[pre[pre[Rank[i]]]];
cilu[i]=fabs(y-hei[i]);
}
}
pre[suc[Rank[i]]]=pre[Rank[i]];
suc[pre[Rank[i]]]=suc[Rank[i]];
}
for (int i=;i<=n;i++) f[i][]=zui[ci[i]],fa[i][]=cilu[i],fb[i][]=zuilu[ci[i]];
for (int j=;j<=;j++)
for (int i=,to=(n-(<<j)+);i<=to;i++)
{
f[i][j]=f[f[i][j-]][j-];
fa[i][j]=fa[i][j-]+fa[f[i][j-]][j-];
fb[i][j]=fb[i][j-]+fb[f[i][j-]][j-];
// cout<<i<<' '<<j-1<<' '<<f[i][j-1]<<' '<<fa[i][j-1]<<' '<<fb[i][j-1]<<endl;
}
// for(int i=1;i<=n;i++,puts("")) for(int j=0;j<=1;j++) printf("[%lld %lld %lld] ",f[i][j],fa[i][j],fb[i][j]);
} void work1()
{
int x0=qread();
bool flag=;
LL ansa=,ansb=;int ans=;
for (int i=;i<=n;i++)
{
LL tota=,totb=;int now=i;
for (int j=;j>=;j--) if (f[now][j])
if (fa[now][j]+fb[now][j]+tota+totb<=x0)
{
tota+=fa[now][j];
totb+=fb[now][j];
now=f[now][j];
// if (i==2) cout<<tota<<' '<<totb<<' '<<now<<endl;
}
if (ci[now] && cilu[now]+tota+totb<=x0) tota+=cilu[now],now=ci[now];
if (!flag) ansa=tota,ansb=totb,flag=,ans=i;
else if (totb)
{
if (!ansb) ansa=tota,ansb=totb,ans=i;
else if (1.0*ansa/ansb>1.0*tota/totb) ansa=tota,ansb=totb,ans=i;
else if (fabs(1.0*ansa/ansb-1.0*tota/totb)<1e- && hei[i]>hei[ans]) ans=i;
}
else if (!ansb && hei[i]>hei[ans]) ans=i;
// cout<<"now"<<now<<endl;
// cout<<tota<<' '<<totb<<endl;
}
printf("%d\n",ans);
} void work2()
{
int m=qread();int x,y;
while (m--)
{
x=qread(),y=qread();
LL tota=,totb=;
for (int j=;j>=;j--) if (f[x][j])
if (tota+fa[x][j]+totb+fb[x][j]<=y)
{
tota+=fa[x][j];
totb+=fb[x][j];
x=f[x][j];
}
if (ci[x] && cilu[x]+tota+totb<=y) tota+=cilu[x];
printf("%lld %lld\n",tota,totb);
}
} int main()
{
n=qread();
for (int i=;i<=n;i++) hei[i]=qread();
prepare();
work1();
work2();
// for (int i=1;i<=n;i++) cout<<zui[i]<<' '<<zuilu[i]<<' '<<ci[i]<<' '<<cilu[i]<<endl;
return ;
}

NOIP2012提高组D1T3 开车旅行的更多相关文章

  1. 【NOIP2012提高组】开车旅行 倍增

    题目分析 朴素的做法就是预处理下一个目的地,然后跑模拟,超时. 本题最重要的考点是倍增优化.设$fa[i][j]$表示a从i出发行驶$2^j$“次”后行驶的路程,$fb[i][j]$表示从i出发行驶$ ...

  2. 【noip 2012】提高组Day1T3.开车旅行

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

  3. NOIP2012提高组

    D1T1.Vigenère密码 模拟 #include<iostream> #include<cstdio> using namespace std; int main() { ...

  4. 刷题总结——疫情控制(NOIP2012提高组)

    题目: 题目背景 NOIP2012 提高组 DAY2 试题. 题目描述 H 国有 n 个城市,这 n 个城市用 n-1 条双向道路相互连通构成一棵树,1 号城市是首都,也是树中的根节点. H 国的首都 ...

  5. GZOJ 1361. 国王游戏【NOIP2012提高组DAY1】

    国王游戏[NOIP2012提高组DAY1] Time Limit:1000MS Memory Limit:128000K Description 国王游戏(game.cpp/c/pas) [问题描述] ...

  6. [NOIP2012] 提高组 洛谷P1081 开车旅行

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

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

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

  8. NOIP2012 提高组 Day 1

    期望得分:100+100+70=270 实际得分:100+50+70=220 T2 没有底 最后剩余时间来不及打高精.思路出现错误 T1 Vigenère 密码 题目描述 16 世纪法国外交家 Bla ...

  9. [NOIP2012] 提高组 洛谷P1084 疫情控制

    题目描述 H 国有 n 个城市,这 n 个城市用 n-1 条双向道路相互连通构成一棵树,1 号城市是首都, 也是树中的根节点. H 国的首都爆发了一种危害性极高的传染病.当局为了控制疫情,不让疫情扩散 ...

随机推荐

  1. pycharm的使用小技巧111

    如果你想快速敲出if __name__ == '__main__':只需你敲个main 然后回车就ok了 import和from xx模块 import *的区别是前者使用时要加模块名加点,后者可以直 ...

  2. [转]强制取消TFS2008中其它成员的签出文件

    本文转自:http://www.cnblogs.com/georgehu/archive/2010/10/23/1859573.html 有个项目,以前的成员离职了,刚好又签出了一个文件在TFS中并且 ...

  3. 实现php间隔一段时间执行一次某段代码

    <?php ignore_user_abort(); //即使Client断开(如关掉浏览器),PHP脚本也可以继续执行.  set_time_limit(0); // 执行时间为无限制,php ...

  4. jq星星评分

    html代码 <div class="make_mark"> <h5>请为这次服务打分</h5> <div class="mar ...

  5. APP崩溃处理

    以前经常遇到APP内部异常情况下的Exception,最初是通过try catch这样的方式处理:但是APP上线后,用户在特地的情况下触发 了某些Exception,当然这些Exception从理论和 ...

  6. jsp 页面获取当前路径

    <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"% ...

  7. 3D旋转矩阵的推导过程

    3D旋转矩阵的推导过程 包含平移的线性变换称作仿射变换,3D中的仿射变换不能用 3 x 3 矩阵表达,必须使用4 x 4矩阵. 一般来说,变换物体相当于以相反的量变换描述这个物体的坐标系.当有多个变换 ...

  8. 大众点评APP分析随笔

    移动APP:大众点评 一.最核心功能:店铺评价功能,用户可以通过此功能对商家进行评分,也可以获取其他人对商家的评分信息. 二.核心功能满足的需求: 1. 去过商家消费的用户:用户已经体验的商家提供的产 ...

  9. 迅为4412全新升级版|3G开发板|4G开发板

    iTOP-Exynos4412开发板采用 Exynos4412的主芯片,具有更高的主频和更丰富外设,配置 2GB 双通道 DDR3的内存及 16GB 存储,支持3G/G模块.GPS模块.陀螺仪.HDM ...

  10. (转)SpringMVC学习(十一)——SpringMVC实现Resultful服务

    http://blog.csdn.net/yerenyuan_pku/article/details/72514034 Restful就是一个资源定位及资源操作的风格,不是标准也不是协议,只是一种风格 ...