传送门

解题思路

这道题刚了一下午,主要就刚在set那里了。先写了一个暴力70分。。之后优化预处理,看着大佬神犇们都用的什么双向链表之类的东西,本蒟蒻不会,又懒得手写平衡树,就拿了个set搞了搞,感觉做麻烦了,我开了两个set,一个存正数一个存负数。因为STL只能求后继,然后每次先求最小,删掉再求次小,再加回来。预处理复杂度应该是nlogn,交了一发75,看了看lyd的书才知道要倍增。f[i][j][0/1]表示一共走了2^i天,在j这个城市,轮A/B开车的到达的城市,如果i=1 f[i][j][k]=f[i-1][f[i-1][j][k]][1-k],意思就是走了2天,一人开了一天,所以由另一个人最后开转移来。如果i>1 f[i][j][k]=f[i-1][f[i-1][j][k]][k], 意思是每个人开了偶数天,所以前后两半的最后都是同一个人开,再预处理一个data表示路程,处理方法和f相近,最后卡了卡常。。

代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<set>
#include<cmath> using namespace std;
const int MAXN = 100005;
typedef long long LL; inline LL rd(){
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)) {if(ch=='-') f=-1;ch=getchar();}
while(isdigit(ch)) {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
return x*f;
} struct Pos{
int id;
LL high;
}pos[MAXN],rev[MAXN]; int n,m,nxt[MAXN][3],x0,st,f[22][MAXN][2];
LL ans[3],dis[MAXN][3],data[22][MAXN][2][2];
double K=1e9;
set<Pos> S,T;
bool operator <(const Pos &A,const Pos &B){
return A.high<B.high;
} int main(){
// freopen("random.txt","r",stdin);
// freopen("A.txt","w",stdout);
n=rd();int lg=log2(n);
for(register int i=1;i<=n;i++) pos[i].high=rd(),pos[i].id=i;
for(register int i=n;i;i--){
rev[i].high=-pos[i].high;rev[i].id=i;
LL mn1=1e18,mn2=1e18;int p1=-1,p2=-1;Pos A;int flag=0;
set<Pos>::iterator it=S.lower_bound(pos[i]);
if(it!=S.end()) {mn1=abs((*it).high-pos[i].high);p1=(*it).id;A=*it;flag=1;}
it=T.upper_bound(rev[i]);
if(it!=T.end() && abs(-((*it).high)-pos[i].high)<=mn1){
mn1=abs(-((*it).high)-pos[i].high);
p1=(*it).id;A=*it;flag=2;
}
Pos B;B.id=A.id;B.high=-A.high;
if(flag==1) S.erase(A),T.erase(B);
else if(flag==2) S.erase(B),T.erase(A);
if(flag){
it=S.lower_bound(pos[i]);
if(it!=S.end()) {mn2=abs((*it).high-pos[i].high);p2=(*it).id;}
it=T.upper_bound(rev[i]);
if(it!=T.end() && abs(-((*it).high)-pos[i].high)<=mn2){
mn2=abs(-((*it).high)-pos[i].high);
p2=(*it).id;
}
}
if(flag==1) S.insert(A),T.insert(B);
else if(flag==2) S.insert(B),T.insert(A);
S.insert(pos[i]);T.insert(rev[i]);
nxt[i][1]=p2,nxt[i][2]=p1;
dis[i][1]=mn2,dis[i][2]=mn1;
// cout<<i<<" "<<nxt[i][2]<<" "<<dis[i][2]<<endl;
f[0][i][0]=nxt[i][1];
f[0][i][1]=nxt[i][2];
data[0][i][0][0]=dis[i][1];
data[0][i][1][0]=0;
data[0][i][0][1]=0;
data[0][i][1][1]=dis[i][2];
}
for(register int i=1;i<=lg;i++)
for(register int j=1;j<=n;j++)
for(register int k=0;k<=1;k++){
if(i==1){
f[1][j][k]=f[0][f[0][j][k]][1-k];
data[1][j][k][0]=data[0][j][k][0]+data[0][f[0][j][k]][1-k][0];
data[1][j][k][1]=data[0][j][k][1]+data[0][f[0][j][k]][1-k][1];
continue;
}
f[i][j][k]=f[i-1][f[i-1][j][k]][k];
data[i][j][k][0]=data[i-1][j][k][0]+data[i-1][f[i-1][j][k]][k][0];
data[i][j][k][1]=data[i-1][j][k][1]+data[i-1][f[i-1][j][k]][k][1];
}
x0=rd();
for(register int i=1;i<=n;i++){
ans[1]=ans[2]=0;int p=i;
for(register int j=lg;j>=0;j--)
if(ans[1]+ans[2]+data[j][p][0][0]+data[j][p][0][1]<=x0){
ans[1]+=data[j][p][0][0];ans[2]+=data[j][p][0][1];
p=f[j][p][0];
}
if(ans[2]==0) {if(K==1e9 && pos[i].high>pos[st].high) st=i;continue;}
if((double)ans[1]/ans[2]<K || ((double)ans[1]/ans[2]==K && pos[i].high>pos[st].high)){
st=i;
K=(double)ans[1]/ans[2];
}
}
printf("%d\n",st);
m=rd();
for(register int i=1;i<=m;i++){
LL x=rd(),y=rd();
ans[1]=ans[2]=0;int p=x;
for(register int j=lg;j>=0;j--)
if(ans[1]+ans[2]+data[j][p][0][0]+data[j][p][0][1]<=y){
ans[1]+=data[j][p][0][0];ans[2]+=data[j][p][0][1];
p=f[j][p][0];
}
printf("%lld %lld\n",ans[1],ans[2]);
}
return 0;
}

LUOGU P1081 开车旅行 (noip 2012)的更多相关文章

  1. 开车旅行 [NOIP 2012]

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

  2. luogu P1081 开车旅行

    传送门 这题的暴力做法显然是照题意模拟,从每个点出发暴力跳.而这个暴跳显然是可以倍增优化的,就是预处理出从每个点,(一开始是A)往后跳\(2^k\)步,能到哪里,以及\(A\)和\(B\)的路程,然后 ...

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

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

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

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

  5. P1081 开车旅行

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

  6. NOIP 2012 洛谷P1081 开车旅行

    Description: 就是两个人开车,只能向东开.向东有n个城市,城市之间的距离为他们的高度差.A,B轮流开车,A喜欢到次近的城市,B喜欢到最近的城市.如果车子开到底了或者车子开的路程已经超过了限 ...

  7. NOIP 2012 P1081 开车旅行

    倍增 这道题最难的应该是预处理... 首先用$set$从后往前预处理出每一个点海拔差绝对值得最大值和次大值 因为当前城市的下标只能变大,对于点$i$,在$set$中二分找出与其值最接近的下标 然后再$ ...

  8. P1081 开车旅行(Not Finish)

    https://www.luogu.org/problemnew/show/P1081

  9. noip2012 P1081 开车旅行

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

随机推荐

  1. HDU 3923 Invoker | 暑训Day1 C题填坑

    暑训第一天,专题为组合数学与概率期望. 最近一个月都没有学习新的知识,上午听聚聚讲课头脑都是一片空白.加上长期没刷题,下午做练习题毫无感觉.到晚上总算理清了蓝书上的一些概念,跟着榜单做题.最后唯独剩下 ...

  2. PAT甲级——【牛客A1005】

    题目描述 Behind the scenes in the computer's memory, color is always talked about as a series of 24 bits ...

  3. mac上安装软件后,桌面上软件的图标如何去掉?

    桌面上的图标是软件的镜像包,默认安装以镜像形式,你选中它,按command+e 就可以推掉它

  4. vim 插入行号

    :let i=1000000|g/^/s//\=i.' '/|let i=i+1

  5. pyinstaller 打包python3.6文件成exe 运行

    1.安装pyinstaller  切换到安装目录下script 运行  如我的目录:F:\Program Files\Python36\Scripts pip  install pyinstaller ...

  6. 利用PHP获取访客IP、地区位置、浏览器及来源页面等信息

    这篇文章主要介绍了利用PHP获取访客IP.地区位置.浏览器及来源页面等信息的相关资料,文中给出了详细的示例代码供大家参考学习,对大家具有一定的参考借鉴价值,需要的朋友们下面来一起看看吧. 前言 本文中 ...

  7. 深入浅出 Java Concurrency (9): 锁机制 part 4[转]

    本小节介绍锁释放Lock.unlock(). Release/TryRelease unlock操作实际上就调用了AQS的release操作,释放持有的锁. public final boolean ...

  8. hibernate 一对一注解

    bi如 用户的阅读历史和文章表是单向一对一关系, 阅读历史中通过deviceId外键关联文章表的主键 然后,再从getter setter上进行注解 @OneToOne(cascade = Casca ...

  9. line-height:2和line-height:2em的区别,它们是有区别的

    line-height:2是2倍的意思,如果内部有不同大小文字的情况下,以最大文字为倍数. line-height:2em也是2倍文字大小的意思,但如果内部有大文字,它还是会以父容 器的大小来计算. ...

  10. 构建支持中文字体的moviepy镜像

    首先是系统的环境问题. linux 安装 moviepy需要很多依赖,安装起来费神费力.配置起来也非常麻烦,最简单的办法是直接使用他人构建好的镜像文件. 再就是字体显示问题. 镜像中的imagmagi ...