Vacation

题目传送门

update(O(n))

看了那个O(n)的方法,感觉自己想的那个O(nlogn)的好傻,awsl。

0车最终通过停车线的时候,状态一定是某个车堵住后面的所有车(这个车也可以是0车)。所以我们要找的就是那个把后面所有都堵住的最前面的车x,x车没有被别的车堵住,从头到尾都按照初始速度行驶,当0车到停车线时,这x车走过的距离d = s_x + 0到x车的车身长度和,时间为d/v_x。

我们假设所有的车都是车x,求出时间,其中最大时间即为答案。因为如正确的时间t = (车i行驶的总路程/车i的平均速度).如果一辆车的速度改变过,只会变小,而用原来的速度算会使答案偏小,而如果这辆车在车x的前面,与x有一段距离,那参与运算的d就偏小了,算出的时间就会偏小。

以下为代码

#include <bits/stdc++.h>
#define INF 0x3f3f3f3f
using namespace std;
typedef long long ll; const int N = 100005; double l[N], s[N], v[N];
double len[N]; int main()
{
int n;
while(scanf("%d", &n) != EOF){
for(int i = 0; i <= n; i ++)
scanf("%lf", &l[i]);
for(int i = 0; i <= n; i ++)
scanf("%lf", &s[i]);
for(int i = 0; i <= n; i ++)
scanf("%lf", &v[i]);
len[0] = 0;
for(int i = 1; i <= n; i ++)
len[i] = len[i - 1] + l[i];
double ans = 0.0;
for(int i = 0; i <= n; i ++)
ans = max(ans, (s[i] + len[i]) / v[i]);
printf("%.10f\n", ans);
}
return 0;
}

解题思路(O(nlog)n)

一开始的时候所有车都是是按照自己原来的速度行驶,直到有一辆车x追上前面的那辆车y,此时的变化只有,1.x的速度变为y的速度2.x后面的车追上x的时间变短。所以我们只要利用优先队列,每一次都找到会在最短的时间内追上前面那辆车的x,不断更新这个过程就好了。

以及实现中的一些细节,1.对于已经追上和被追上的两辆车,我们可以看成一辆火车,每辆车就是一节车厢,其速度就是最前面那节车厢的速度,当一辆火车追上另一辆的时候,新的火车头就是前面被追上车厢的火车头,新的火车尾就是后面火车的火车尾,所以我们记录每节作为火车头的车厢对应的火车尾,和每节火车尾对应的火车头。2.优先队列中的时间指是第几秒会追上前面的车。由于前面的车会变速,所以要记录上一次变速的时间,以及上一次变速前已经走了多少或还剩下多少距离。3.只有后面的车可以追上前面的车时才进队。

还有就是,不要直接memset,数据组数很多,会超时。

代码如下

#include <bits/stdc++.h>
#define INF 0x3f3f3f3f
using namespace std;
typedef long long ll; const int N = 100005; double len[N], s[N], v[N];
int l[N], r[N]; struct T{
double t;
int x;
T(int x, double t): x(x), t(t){}
bool operator<(const T& a)const{
return t > a.t;
}
}; bool vis[N];
double t[N], dis[N]; int main()
{
int n;
while(scanf("%d", &n) != EOF){
for(int i = 0; i <= n; i ++)
scanf("%lf", &len[i]); //车长
for(int i = 0; i <= n; i ++)
scanf("%lf", &s[i]);
for(int i = 0; i <= n; i ++)
scanf("%lf", &v[i]);
for(int i = 0; i <= n; i ++)
l[i] = r[i] = i;
priority_queue<T> pq;
for(int i = 0; i < n; i ++){
t[i] = 0.0;
vis[i] = false;
dis[i] = s[i] - s[i + 1] - len[i + 1];
if(v[i] > v[i + 1])
pq.push(T(i, dis[i] / (v[i] - v[i + 1])));
}
double d = 0.0; //0号车已走的路程
double last = 0.0;
while(!pq.empty()){
T top = pq.top();
pq.pop();
if(vis[top.x])
continue;
vis[top.x] = true;
double time = top.t;
int x = top.x;
if(d + v[r[0]] * (time - last) >= s[0])
break;
d += v[r[0]] * (time - last);
r[l[x]] = r[x + 1];
l[r[x + 1]] = l[x];
if(l[x] != 0){
dis[l[x] - 1] -= (time - t[l[x] - 1]) * (v[l[x] - 1] - v[x]);
t[l[x] - 1] = time;
if(v[l[x] - 1] > v[r[x + 1]])
pq.push(T(l[x] - 1, time + dis[l[x] - 1] / (v[l[x] - 1] - v[r[x + 1]])));
}
last = time;
}
printf("%.10f\n", last + (s[0] - d) / v[r[0]]);
}
return 0;
}

最新文章

  1. Java学习笔记(未完待续)
  2. 2.7---判断链表是否是回文(CC150)
  3. 初识python第一天
  4. Apache Spark源码走读之14 -- Graphx实现剖析
  5. android记住密码和自动登陆
  6. http协议分析工具
  7. Multi-bit per cell storage
  8. hdu 1757 A Simple Math Problem(矩阵快速幂乘法)
  9. linux之文本编辑器
  10. Web面试之JQuery
  11. loadrunner controller:实时查看VUser的运行情况
  12. 【hibernate初探】之接口说明,session使用
  13. Hibernate第十一篇【配置C3P0数据库连接池、线程Session】
  14. 解决mariadb grant ERROR 1045 (28000): Access denied for user
  15. Tarjan学习笔记
  16. SaltStack 数据系统 Grains Pillar
  17. LeetCode--111--最长公共前缀
  18. NTP服务器配置
  19. Office for Mac
  20. 敏捷软件开发:原则、模式与实践——第13章 写给C#程序员的UML概述

热门文章

  1. OpenSSL包括了8个功能
  2. HTML续
  3. redis的下载及使用
  4. java集合框架collection(4)HashMap和Hashtable的区别
  5. webpack 编译ES6
  6. c++汉诺塔相关知识总结1
  7. 5个现在就该使用的数组Array方法: indexOf/filter/forEach/map/reduce详解(转)
  8. 分享Sql Server 2008 r2 数据备份,同步服务器数据(二.本地发布,订阅)
  9. 对def函数的参数认识
  10. 使用Core Audio实现VoIP通用音频模块