博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
NOIP2011 观光公交 加强版
阅读量:5334 次
发布时间:2019-06-15

本文共 2481 字,大约阅读时间需要 8 分钟。

题目大意

给定从左到右的$n$个车站以及两两之间通行的需要的时间。

有$m$个人,第$i$个人会在$T_i$时刻出现在$a_i$车站,目的地是$b_i$。

一辆车第$0$时刻出现在一号站台,从左向右驶去,每经过一个车站(包括$1$号),它会等待知道所有应该在该站台出现的乘客都出现才会继续驶向下一个站台。

定义一个人花费的时间是列车到达$b_i$的时刻$-T_i$。你有$k$次机会使得某相邻两站花费的时间减少一个时间单位(需要始终保证非负),求这$m$个人花费时间之和的最小值。

 

题解

在$n,m,k$均很小的情况下可以用各种算法搞过去,但加强后不得不优化复杂度。

前置技能:$O(nk)$的贪心()

先设一个最终答案为$ans$,初始时$ans=-\sum T_i$

仍是考虑设第$i$个车站最后一个人到达时间为$G_i$,到达$i$的时间比$G_i$多出了$S_i$的时间,第$i$个车站到第$i+1$个车站的距离为$D_i$。

那么$S_i=\max\{S_{i-1}+G_{i-1}+D_{i-1}-G_{i}\},ans+=S_i+G_i$

考虑一段区间$[L,R]$,若有$(L,R]$内的$S$均大于$0$,那么减少$D_L K$个单位可以使得所有$(L,R]$的到达时间减少$K$,那么最终答案减少的就是$K\times$区间内是终点的数量。

考虑用线段树维护$S$,将每一个有意义的$[L,R]$用一个大根堆存起来(以区间内终点数量为关键字)。

每次取出堆顶$[L,R]$,求$x=\min\{D_L,\min\{S_i\}i\in(L,R]\}$即为区间内可以减少的时间,那么将$k$次机会中的$x$个用与减少区间答案。

接下来一定会至少出现一个$i$使得$S_i=0$或$k=0$或$D_L=0$,那么原来有意义的区间至多分裂成两个有意义的区间,在分别丢进堆里不断处理即可。

可见,该算法的复杂度及基本与$k$无关,甚至与$m$无关。

复杂度仅为有意义的区间数量(不超过$2n$个)在带一个线段树的和堆的$log$,大概就是$O(n\log n)$。

 

#include
#define LL long long#define M 600020using namespace std;namespace IO{ const int BS=(1<<20)+5; char Buffer[BS],*HD,*TL; char Getchar(){if(HD==TL){TL=(HD=Buffer)+fread(Buffer,1,BS,stdin);} return (HD==TL)?EOF:*HD++;} int read(){ int nm=0,fh=1; char cw=Getchar(); for(;!isdigit(cw);cw=Getchar()) if(cw=='-') fh=-fh; for(;isdigit(cw);cw=Getchar()) nm=nm*10+(cw-'0'); return nm*fh; }}using namespace IO;int n,m,p[M<<2],pos[M<<2],tg[M<<2],val[M],D[M],K,last;LL ans,gt[M],st[M];#define pushup int sn=(p[x<<1]
<<1|1])?(x<<1):(x<<1|1);pos[x]=pos[sn],p[x]=p[sn]struct seg{ int LS,RS;seg(){} seg(int _LS,int _RS){LS=_LS,RS=_RS;} bool operator<(const seg&ot)const{return val[RS]-val[LS]
H;void build(int x,int l,int r){ if(l==r){pos[x]=r,p[x]=st[r];return;} int mid=((l+r)>>1); build(x<<1,l,mid); build(x<<1|1,mid+1,r); pushup;}#define pushdown tg[x<<1]+=tg[x],tg[x<<1|1]+=tg[x],p[x<<1]-=tg[x],p[x<<1|1]-=tg[x],tg[x]=0void mdf(int x,int l,int r,int ls,int rs,int dt){ if(ls<=l&&r<=rs){tg[x]+=dt,p[x]-=dt;return;} if(r
>1); pushdown; mdf(x<<1,l,mid,ls,rs,dt),mdf(x<<1|1,mid+1,r,ls,rs,dt); pushup;}int getnode(int x,int l,int r,int ls,int rs){ if(ls<=l&&r<=rs) return x;if(r
>1); pushdown; int t1=getnode(x<<1,l,mid,ls,rs); int t2=getnode(x<<1|1,mid+1,r,ls,rs); return p[t1]>p[t2]?t2:t1;}int main(){ n=read(),m=read(),K=read(),last=1,p[0]=1000000000; for(int i=1;i
0&&!H.empty()){ seg tmp=H.top();H.pop();int ls=tmp.LS,rs=tmp.RS,k=0; if(ls+1

 

转载于:https://www.cnblogs.com/OYJason/p/9910831.html

你可能感兴趣的文章
python基础学习笔记——反射
查看>>
CS231n笔记 Lecture 5 Convolutional Neural Networks
查看>>
Oracle游标
查看>>
详解nginx 配置多个tomcat共用80端口
查看>>
浅谈欧拉定理的证明
查看>>
MySQL查询机制
查看>>
JAVA中哪些情况下类不能够被继承?
查看>>
【编程之美】2.13 子数组的最大乘积
查看>>
Collection2
查看>>
62. Unique Paths
查看>>
C++普通函数与模板函数以及特化函数重载的优先级问题
查看>>
php 经典分页
查看>>
JavaScript 中的面向对象的初步认识
查看>>
mybaits中"#"和"$"的区别
查看>>
黑马程序猿——12,多线程(2)
查看>>
2.5 使用git对项目进行版本控制
查看>>
windows phone textblock C#设置颜色以及换行
查看>>
Windows Phone开发(29):隔离存储C 转:http://blog.csdn.net/tcjiaan/article/details/7447469...
查看>>
Windows Phone开发(6):处理屏幕方向的改变 转:http://blog.csdn.net/tcjiaan/article/details/7273107...
查看>>
LockBits in GDI+【转】http://timothyqiu.com/archives/lockbits-in-gdiplus/
查看>>