传送门

解题思路:

  一道比较有趣的题,解题工作主要分为两块:

  ①找出k(k=0表示小A先走,k=1表示小B先走,下面同理)点i出发下一个到达的点to[k][i];

  一开始偷懒用了vector(偷懒一时爽),由于vector的erase操作是o(n)的,这个预处理时间复杂度就彪到o(n2)了。这里改成set就可以将复杂度降到o(nlogn),用链表的话讲道理可以降到o(n)但是排序就要o(nlogn)毫无*用。具体操作就是二分再前后比较(简称瞎搞)。

  ②两个人点i出发走2j步走各自开车所走过的路程分别表示为da[i][j][k],db[i][j][k],最终达到点表示为f[i][j][k]。da[i][j][k]表示k从i点出发走了2j步的小A要开车的路程。那么da[i][j][k]+db[i][j][k]就表示了k先手开车从点i出发走了2j步共走过的路程。预处理出第一步和第二步,(转移方程见代码)。

  运用二进制拆分的思想,假设出发点为s,p表示当前到达的点,ta、tb分别表示小A和小B从s出发达到p点各自走过的路程,x表示能走得最大路程。从log(最大步数)到0遍历j,如果ta+tb+da[p][j][k]+db[p][j][k]<=x,则令ta和tb分别加上da[p][j][k]和db[p][j][k],p=f[i][j][k]。枚举所有的点进行上述操作就可得到结果。

丑陋的代码:

#include<bits/stdc++.h>
using namespace std;
const int maxn=1e5+;
typedef long long ll;
ll da[maxn][][],db[maxn][][];map <int,int> m;
int h[maxn],to[][maxn],f[maxn][][];set <int> v;
int main()
{
int n;scanf("%d",&n);
for(int i=;i<=n;i++)
scanf("%d",&h[i]),v.insert(h[i]),m[h[i]]=i;
for(int i=;i<n;i++){
v.erase(h[i]);
auto id=v.lower_bound(h[i]);auto id1=id,id2=id;
if(id==v.begin()){
to[][i]=m[(*id)];
id1++;
if(id1!=v.end())
to[][i]=m[(*id1)];
}
else if(id==v.end()){
id1--;
if(id!=v.begin())
to[][i]=m[(*id1)];
if(id1!=v.begin())
id1--,to[][i]=m[(*id1)];
}
else{
id1--;
int t1=abs(h[i]-(*id)),t2=abs(h[i]-(*id1));
if(t1>t2){
to[][i]=m[(*id1)];id2=id1;id2--;
int t3=(id1!=v.begin())?abs(h[i]-(*id2)):2e9;
if(t1>=t3) to[][i]=m[(*id2)];
else if(t1<t3) to[][i]=m[(*id)];
}
else if(t1<t2){
to[][i]=m[(*id)];id2++;
int t3=(id2!=v.end())?abs(h[i]-(*id2)):2e9;
if(t2>t3) to[][i]=m[(*id2)];
else if(t2<=t3) to[][i]=m[(*id1)];
}
else{
to[][i]=m[(*id1)];
to[][i]=m[(*id)];
}
}
}
for(int i=;i<=n;i++)
f[i][][]=to[][i],f[i][][]=to[][i];
for(int i=;i<=n;i++)
for(int k=;k<=;k++)
f[i][][k]=f[f[i][][k]][][-k];
for(int j=;j<=;j++)
for(int i=;i<=n;i++)
for(int k=;k<=;k++)
f[i][j][k]=f[f[i][j-][k]][j-][k];
for(int i=;i<=n;i++)
da[i][][]=abs(h[(to[][i]==)?i:to[][i]]-h[i]),
db[i][][]=abs(h[(to[][i]==)?i:to[][i]]-h[i]);
for(int i=;i<=n;i++)
for(int k=;k<=;k++)
da[i][][k]=da[i][][k]+da[f[i][][k]][][-k],
db[i][][k]=db[i][][k]+db[f[i][][k]][][-k];
for(int j=;j<=;j++)
for(int i=;i<=n;i++)
for(int k=;k<=;k++)
da[i][j][k]=da[i][j-][k]+da[f[i][j-][k]][j-][k],
db[i][j][k]=db[i][j-][k]+db[f[i][j-][k]][j-][k];
int x;scanf("%d",&x);int ans;double mi=1e18;
for(int s=;s<n;s++){
int p=s;ll ta=,tb=;
for(int j=;j>=;j--)
if(ta+tb+da[p][j][]+db[p][j][]<=x){
ta+=da[p][j][];
tb+=db[p][j][];
p=f[p][j][];
}
double t;if(tb==) t=1e18-;
else t=(double)1.0*ta/tb;
if(t<mi) ans=s,mi=t;
}
printf("%d\n",ans);
int q;scanf("%d",&q);
while(q--){
int s;scanf("%d%d",&s,&x);
int p=s;ll ta=,tb=;
for(int j=;j>=;j--)
if(ta+tb+da[p][j][]+db[p][j][]<=x){
ta+=da[p][j][];
tb+=db[p][j][];
p=f[p][j][];
}
printf("%d %d\n",ta,tb);
}
}

CH5701 开车旅行(倍增dp+set)的更多相关文章

  1. CH5701 开车旅行

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

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

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

  3. $Noip2012\ Luogu1081$ 开车旅行 倍增优化$ DP$

    Luogu Description Sol 1.发现对于每个城市,小A和小B的选择是固定的,可以预处理出来,分别记为ga[],gb[] 2.并且,只要知道了出发城市和出发天数,那么当前城市和小A,小B ...

  4. 【vijos1780】【NOIP2012】开车旅行 倍增

    题目描述 有\(n\)个城市,第\(i\)个城市的海拔为\(h_i\)且这\(n\)个城市的海拔互不相同.编号比较大的城市在东边.两个城市\(i,j\)之间的距离为\(|h_i-h_j|\) 小A和小 ...

  5. P1081 开车旅行[倍增](毒瘤题)

    其实就是个大模拟. 首先,根据题意,小A和小B从任意一个城市开始走,无论\(X\)如何,其路径是一定唯一的. 显然对于两问都可以想出一个\(O(n^2)\)的暴力,即直接一步一步地向右走. 首先,我们 ...

  6. 洛谷1081 (NOIp2012) 开车旅行——倍增预处理

    题目:https://www.luogu.org/problemnew/show/P1081 预处理从每个点开始a能走多少.b能走多少.可以像dp一样从后往前推. 但有X的限制.所以该数组可以变成倍增 ...

  7. Luogu1081 NOIP2012 开车旅行 倍增

    题目传送门 为什么NOIP的题目都这么长qwq 话说2012的D1T3和D2T3都是大火题啊qwq 预处理神题 对于这种跳跳跳的题目考虑使用倍增优化枚举.先预处理某个点之后距离最小和次小的城市,然后倍 ...

  8. 洛谷P1081 开车旅行(倍增)

    题意 题目链接 Sol 咕了一年的题解.. 并不算是很难,只是代码有点毒瘤 \(f[i][j]\)表示从\(i\)号节点出发走了\(2^j\)轮后总的距离 \(da[i][j]\)同理表示\(a\)的 ...

  9. 洛谷 P1081 开车旅行 —— 倍增

    题目:https://www.luogu.org/problemnew/show/P1081 真是倍增好题! 预处理:f[i][j] 表示从 i 点开始走 2^j 次 AB (A,B各走一次)到达的点 ...

随机推荐

  1. MySQL InnoDB表的碎片量化和整理(data free能否用来衡量碎片?)

    网络上有很多MySQL表碎片整理的问题,大多数是通过demo一个表然后参考data free来进行碎片整理,这种方式对myisam引擎或者其他引擎可能有效(本人没有做详细的测试).对Innodb引擎是 ...

  2. 我用STM32MP1做了个疫情监控平台3—疫情监控平台实现

    目录 1.前言 2.数据接口的获取 3.Qt界面的实现 4.在开发板上运行Qt程序 5.使用无线模块联网 6.代码下载 @ 1.前言 之前我使用桌面版本Qt实现了肺炎疫情监控平台:基于Qt的新冠肺炎疫 ...

  3. WebRTC的RTCPeerConnection()原理探析

    从getUserMedia()到RTCPeerConnection(),自认为难度陡增.我想一方面是之前在Linux平台上学习ROS调用摄像头时,对底层的外设接口调用.摄像头参数都有学习理解:另一方面 ...

  4. WSGI标准、MVC和MTC框架

    WSGI服务: wsgiref模块其实就是将整个请求信息给封装了起来,就不需要你自己处理了,假如它将所有请求信息封装成了一个叫做request的对象,那么你直接request.path就能获取到用户这 ...

  5. Chrome80调整SameSite策略对IdentityServer4的影响以及处理方案(翻译)

    首先,好消息是Goole将于2020年2月份发布Chrome 80版本.本次发布将推进Google的"渐进改良Cookie"策略,打造一个更为安全和保障用户隐私的网络环境. 坏消息 ...

  6. PhpStorm+XAMPP+Xdebug 集成开发和断点调试环境配置

    0x01 Xdebug安装 参考:https://xdebug.org/docs/install cd xdebug-/ phpize sudo ./configure --enable-xdebug ...

  7. HTTP 错误 500.21 模块 IIS Web Core

    如果出现如下图错误 就是iis没有安装Web Core模块 下载并安装DotNetCore.1.0.4_1.1.1-WindowsHosting.exe 安装完成之后会出现如下 证明安装成功.网站也可 ...

  8. 贪心-谷歌-857. 雇佣 K 名工人的最低成本

    2020-03-15 22:00:39 问题描述: 有 N 名工人. 第 i 名工人的工作质量为 quality[i] ,其最低期望工资为 wage[i] . 现在我们想雇佣 K 名工人组成一个工资组 ...

  9. 使用PyTorch建立图像分类模型

    概述 在PyTorch中构建自己的卷积神经网络(CNN)的实践教程 我们将研究一个图像分类问题--CNN的一个经典和广泛使用的应用 我们将以实用的格式介绍深度学习概念 介绍 我被神经网络的力量和能力所 ...

  10. 巴什博弈 HDU-1846

    描述:一堆石子有 n 个 ,两个人开始轮流取,每人最多取m个,最少取1个,最后一个将石子取完的是赢家. 思路:对于先手来说,如果有(m+1)个石子,先手取 k 个,后手就可以取 m+1-k 个,所以有 ...