年轻人要讲抄德,抄袭作业又蠢又坏,大家耗子尾汁


bupt python 排队前进问题

  • 题目描述
  • 输入
  • 输出
  • 样例输入 Copy
  • 样例输出 Copy
  • 思路
  • 模拟运动方式
  • 实现代码
  • 等价转化为短板效应
  • 实现代码


题目描述

有 n 个人排队向一个方向前进,他们前进的速度并不一定相同。 最开始即 t=0 时,每个人的位置并不相同。可以把他们放在数轴上,设他们前进的方向为正方向,对于从左往右第 i 个人,编号为 i,他的初始位置为xi ,初始速度为vi。编号为1的人(队尾,位于数轴最左侧)的位置总为坐标原点,即总有x1=0。(位置单位为米,速度单位为米每秒)。 虽然他们的前进速度不同,但是他们要保证前后顺序不能变。即i追赶上 i+1 的时候, i 将会紧跟 i+1 以 i+1 的当前速度前进.可以认为他们是紧挨着的,之间的距离可以忽略不计。 求编号为1的人前进 s 米需要多少秒.

输入

第一行两个整数n,s,其中(1<=n,s<=100,000),代表n个人排队前进,以及最后的一个人需要前进的距离为s米。 接下来n行,每行两个整数xi,vi,代表第i个人的位置xi,以及他的初始速度vi,保证(0=x1≤x2≤…≤xn≤100,000,1≤vi≤100,000)。

输出

输出一个小数,按照四舍五入的原则恰好保留小数点后两位(测试数据保证答案的小数点后第三位不是4或5)。

样例输入 Copy

3 4
0 3
1 2
2 1

样例输出 Copy

2.00

思路

思路主要有两种:

模拟运动方式

这种方法就是根据题目一五一十对所有人的运动过程进行模拟,主要过程是不断遍历所有人,在1号人物到达s路程之前,每一轮都尝试找到“需要最少时间追上其前面的人”的人。那么每轮扫描都会将时间向前推进这一段“用来追赶的最小时间”,直到在某一轮扫描后经过“这段时间”能够让1号人物到达s路程

实现代码

n, s = map(int, input().split())
xv = []
for i in range(n):
    xi,vi = map(int,input().split(' '))
    xv.append([xi,vi])
time = []                                           # 用于扫描一轮计算相邻两人的追赶时间
dou = 0.0
result = 0.0                                        # 计算结果
change = []
while not xv[0][0] == s:
    change.clear()
    time.clear()    
    for i in range(len(xv) - 1):
        f_speed = float((xv[i][1] - xv[i + 1][1]))  # 两个人之间的相对速度
        if f_speed > 0:                             # 后者可以追上前者
            time.append((xv[i + 1][0] - xv[i][0]) / f_speed)
        else:                                       # 后者肯定追不上前者
            time.append(1000000000)
    mt = min(time)                                  # 取这一轮中追赶所需的最短时间
    flag = bool(False)
    for j in range(len(xv)):
        if j == 0:                                  # 如果在这一轮扫描后,这段时间1号能够到达s
            if xv[j][1] * mt + xv[j][0] >= s:
                result += float((s - xv[j][0]) / xv[j][1])
                xv[j][0] = s
                flag = True
                break
        xv[j][0] += float(xv[j][1] * mt)
    if flag:
        break
    result += float(mt)
    cnt = time.count(mt)                            # 因为经过这一段最下时间可能有不少人都同时追上了前面的人
    start = 0                                       
    while cnt > 0:                                  # 真的有至少两个人都同时追上了其前面的人
        start1 = time.index(mt, start)              # 找出具有最短时间值进行追赶的对象的索引位置
        change.append(start1)                       # change记录了这些索引位置,后续会回到xv改速度和位置
        start = start1 + 1
        cnt -= 1
    # 对于追赶上的两个人,合并成一个就行
    for i in range(len(change) - 1, -1, -1):
        for j in range(change[i], -1, -1):
            if xv[j][0] == xv[j + 1][0]:
                xv[j][1] = xv[j + 1][1]
    if xv[0][0] == xv[1][0] and xv[0][1] == xv[1][1]:
        xv.pop(0)
print("%.2f" % result)

运行结果不论怎么优化都无法避免超时(>3000ms)。经过分析,最差情况下,是可能需要所有队伍后面的人都追上前面的人才行的,也就是说,while循环这时足足需要O(n)级别的遍历次数。我们需要注意,每次while循环里面都还有三个是O(n)级别的for循环。所以这种算法在最差情况下的时间复杂度足有O(n2)。因此这种算法由于思路较为基础,并不算是非常不错的算法。

等价转化为短板效应

因为不管你走的多快,都可能最终因为追上前面那个人而不得不放慢脚步。这就像木桶短板效应一样,木桶装的水取决于短板,这个排队问题中整个队伍(或局部队伍)的前进速率也取决于速度最慢的那个人。
所以对于初始位置就小于s的所有人中,就算有某些人初始速度飞快,也会因为他前面某些龟速人而不得不和他保持同步,从而导致飞速人的运行过程和龟速人的运行过程变得实质相同。而1号人物让这个问题更加简单,因为他就是队尾那位,所以只需要找出初始位置在0-s这个区间里走到s位置最晚的那个人就行,因为不管他是哪个人,都有可能让1追上并限制1的速度,从而让1和自己的龟速运动过程等价。

实现代码

n,s = map(int,input().split(' '))
xv = []
for i in range(n):
    xi,vi = map(int,input().split(' '))
    xv.append([xi,vi])
# 找出0-s中最接近s的那个人的位置
for i in reversed(range(n)):
    if xv[i][0]<s:
        slowest = i
        break
# 在所有s点之前的人中,找出最慢的那个
t_min = (s-xv[slowest][0]) / xv[slowest][1]
for i in reversed(range(slowest)):
    temp = (s-xv[i][0]) / xv[i][1]
    if temp > t_min:
        t_min = temp
print("%.2f" % t_min)

本算法只有三个独立for循环,而且都是O(n)级别的。该算法复杂度在O(n),最终放在OJ上跑,结果也没有超时,问题解决。