传送门

解题思路:

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

  ①找出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. vuepress-theme-reco + Github Actions 构建静态博客,部署到第三方服务器

    最新博客链接 Github链接 查看此文档前应先了解,vuepress基本操作 参考官方文档进行配置: vuepress-theme-reco VuePress SamKirkland / FTP-D ...

  2. VMWARE虚拟机安装系统提示CPU已被客户机操作系统禁用和secureCUT乱码

    错误:VMWARE虚拟机安装系统提示CPU已被客户机操作系统禁用 改正:找到虚拟机的位置找到下图灰色的部分:打开 .vmx后缀的操作系统配置文件,加入以下代码: cpuid.1.eax = :: 2. ...

  3. jsp(3,6,9) EL表达式及JSTL

    1. jsp 1.1jsp是什么 全称: Java Server Pages,java服务器页面.和Servlet一样,是sun公司定义的一种动态网页开发技术.    特点:基于html模版,可以在h ...

  4. xheditor图片上传

    前端步骤:分为三部,这三部做完之后就能正确显示富文本了 1.下载xheditor文件,并按照如下要求进行引入: <!-- xheditor富文本的文件引入 --> <script t ...

  5. cmdb简介

    目录: 1.为啥要做cmdb

  6. hdu2066多源最短路

    题目链接:http://icpc.njust.edu.cn/Problem/Hdu/2066/ SPFA可以高效过,代码如下: #include<bits/stdc++.h> using ...

  7. selenium 使用教程详解-java版本

    第一章 Selenium 概述 1.1.Selenium 发展史 ​ Selenium是一系列基于Web的自动化工具,提供一套测试函数,用于支持Web自动化测试.函数非常灵活,能够完成界面元素定位.窗 ...

  8. Android之MVC、MVP、MVVM

    本文将详细阐述以下MVC.MVP.MVVM三种理念的定义 MVC MVC全名是Model View Controller,是软件工程中的一种软件架构模式,把软件系统分为三个 基本部分:模型(Model ...

  9. 推荐系统(Recommender Systems)

    本博客是针对Andrew Ng在Coursera上的machine learning课程的学习笔记. 目录 基于内容的推荐(Content-based recommendation) 问题表述 问题范 ...

  10. ​知识图谱与机器学习 | KG入门 -- Part1 Data Fabric

    介绍 如果你在网上搜索机器学习,你会找到大约20500万个结果.确实是这样,但是要找到适合每个用例的描述或定义并不容易,然而会有一些非常棒的描述或定义.在这里,我将提出机器学习的另一种定义,重点介绍一 ...