CH5701 开车旅行(倍增dp+set)
传送门
解题思路:
一道比较有趣的题,解题工作主要分为两块:
①找出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)的更多相关文章
- CH5701 开车旅行
题意 5701 开车旅行 0x50「动态规划」例题 描述 小A和小B决定利用假期外出旅行,他们将想去的城市从1到N编号,且编号较小的城市在编号较大的城市的西边,已知各个城市的海拔高度互不相同,记城市 ...
- P1081 [NOIP2012]开车旅行[倍增]
P1081 开车旅行 题面较为啰嗦.大概概括:一个数列,只能从一个点向后走,两种方案:A.走到和自己差的绝对值次小的点B.走到和自己差的绝对值最小点:花费为此差绝对值:若干询问从规定点向后最多花 ...
- $Noip2012\ Luogu1081$ 开车旅行 倍增优化$ DP$
Luogu Description Sol 1.发现对于每个城市,小A和小B的选择是固定的,可以预处理出来,分别记为ga[],gb[] 2.并且,只要知道了出发城市和出发天数,那么当前城市和小A,小B ...
- 【vijos1780】【NOIP2012】开车旅行 倍增
题目描述 有\(n\)个城市,第\(i\)个城市的海拔为\(h_i\)且这\(n\)个城市的海拔互不相同.编号比较大的城市在东边.两个城市\(i,j\)之间的距离为\(|h_i-h_j|\) 小A和小 ...
- P1081 开车旅行[倍增](毒瘤题)
其实就是个大模拟. 首先,根据题意,小A和小B从任意一个城市开始走,无论\(X\)如何,其路径是一定唯一的. 显然对于两问都可以想出一个\(O(n^2)\)的暴力,即直接一步一步地向右走. 首先,我们 ...
- 洛谷1081 (NOIp2012) 开车旅行——倍增预处理
题目:https://www.luogu.org/problemnew/show/P1081 预处理从每个点开始a能走多少.b能走多少.可以像dp一样从后往前推. 但有X的限制.所以该数组可以变成倍增 ...
- Luogu1081 NOIP2012 开车旅行 倍增
题目传送门 为什么NOIP的题目都这么长qwq 话说2012的D1T3和D2T3都是大火题啊qwq 预处理神题 对于这种跳跳跳的题目考虑使用倍增优化枚举.先预处理某个点之后距离最小和次小的城市,然后倍 ...
- 洛谷P1081 开车旅行(倍增)
题意 题目链接 Sol 咕了一年的题解.. 并不算是很难,只是代码有点毒瘤 \(f[i][j]\)表示从\(i\)号节点出发走了\(2^j\)轮后总的距离 \(da[i][j]\)同理表示\(a\)的 ...
- 洛谷 P1081 开车旅行 —— 倍增
题目:https://www.luogu.org/problemnew/show/P1081 真是倍增好题! 预处理:f[i][j] 表示从 i 点开始走 2^j 次 AB (A,B各走一次)到达的点 ...
随机推荐
- 有关EPX Studio使用DELPHI5作为基础环境版本的说明
英巴卡迪诺北京科技有限公司,地址是北京市朝阳门外大街18号丰联广场B座813B,这家公司这家公司不拥有:delphi 1.0~delphi7.0 .delphi 2005版本的著作权,这些都还是属于B ...
- 网络安全从入门到精通(第一章-1)Web服务器通信原理
本文内容 IP地址 域名 端口 HTTP协议 从访客角度看网页浏览器流程 常见服务器系统 路径 Web容器 常见的Web容器 !!!多动手,多动手,只看只听是不行的!!! 1,IP地址:就是计算机在互 ...
- CVE-2019-3396:Confluence未授权模板注入_代码执行
title: Confluence未授权模板注入/代码执行(CVE-2019-3396) tags: [poc,cve] 简介 Confluence是一个专业的企业知识管理与协同软件,也可以用于构建企 ...
- 33. CentOS7 静态ip设置
1.网络连接选择NAT模式: 2.关闭vmware的dhcp:选择编辑-->虚拟网络编辑器,选择VMnet8,去掉使用本地DHCP服务将ip地址分配给虚拟机(D). 3. 点击NAT设置(S)查 ...
- Django实现简单的用户添加、删除、修改等功能
一. Django必要的知识点补充 1. templates和static文件夹及其配置 1.1 templates文件夹 所有的HTML文件默认都放在templates文件夹下. 1.2 stati ...
- 太赞了!阿里几位工程师重写了 《Java 并发编程》
事情是这样的,前些日子和得知一个读者在准备阿里的面试,我蛮有兴趣的跟他聊了起来,随着话题越来越深入,我发现这位读者有意思,他和几位阿里的工程师之前编写了一本 concurrent.redspider. ...
- java split方法使用注意事项
在java.lang包中有String.split()方法,返回是一个数组. 使用时要注意参数如果是特殊符号的话要进行转义. 1."."和"|"都是转义字符,必 ...
- 大型Java进阶专题(五) 设计模式之单例模式与原型模式
前言 今天开始我们专题的第四课了,最近公司项目忙,没时间写,今天抽空继续.上篇文章对工厂模式进行了详细的讲解,想必大家对设计模式合理运用的好处深有感触.本章节将介绍:单例模式与原型模式.本章节参考 ...
- 主从校验工具pt-table-checksum和pt-table-sync工作原理
pt-table-checksum和pt-table-sync是常用来做MySQL主从数据一致性校验的工具,pt-table-checksum只校验数据,不能对数据进行同步:pt-table-sync ...
- display:flex 简单记录
1.有写了 display:flex:这个就是 采用了 flex布局的 元素 这个元素可以 写 6个属性: flex-direction : row | column | row-reverse ...