[SMOJ1854]加油站

这题好像是这么多次模拟赛以来相对比较简单的一题,但其中包含的解题思想也是值得重视的,而且实现上也有一些细节需要注意。

直接考虑“经过某加油站时是否加油”不好做,因为我们无从知道,现在先加了油,会不会比较吃亏。
因此需要稍微变换一下思考方式。在汽车开往小镇的途中,只有在加油站才可以加油。但是,如果认为“在到达加油站时,就获得了一次在之后的任何时候都可以加 Fi 单位汽油的权利”,在解决问题上应该也是一样的。而在之后需要加油时,就认为是在之前经过的加油站加的油就可以了。

那么,因为希望到达终点时加油次数尽可能少,所以当燃料为 0 了之后再进行加油看上去是一个不错的方法。在燃料为 0 时,应该使用哪个加油站来加油呢?显然,应该选能加油量 Fi 最大的加油站。
为了高效地进行上述操作,我们可以使用堆、线段树或优先队列等数据结构进行维护,使用从大到小的顺序依次取出数值。
也就是说:

  • 在经过加油站时,往数据结构中加入 Fi
  • 当燃料箱空了时
    • 如果数据结构中没有可取值,则无法到达终点
    • 否则取出数据结构中的最大元素,并用来给汽车加油

需要注意的细节有:

  • 给的距离不一定是按顺序排列的,因此需要先排序(Monad 就这样丢了 75 分)
  • 题目给出的是“加油点位于距离小镇 Di(1Di<L) 单位距离的地方”,而不是出发点

算法总的时间复杂度为 O(nlog2n) ,空间复杂度为 O(n)
参考代码:

//1854.cpp
#include 
#include 
#include 
#include 
#include 
#include 

using namespace std;

const int MAXN = 5e4 + 100;

struct Tstop {
    int dis, amount;
    bool operator < (const Tstop x) const { return dis < x.dis; }
} stop[MAXN];

int N, L, P;

int solve() {
    priority_queue  pq; int ans = 0; //维护加油站的优先队列
    while (!pq.empty()) pq.pop();
    int last = 0; // ans:加油次数,last:现在所在位置,P:油箱中汽油的量
    for (int i = 0; i <= N; i++) {
        int d = stop[i].dis - last; //接下去要前进的距离
        while (P < d) { //不断加油直到油量足够行驶到下一个加油站
            if (pq.empty()) return -1;
            P += pq.top(); pq.pop();
            ++ans;
        }
        P -= d;
        last = stop[i].dis;
        pq.push(stop[i].amount);
    }
    return ans;
}

int main(void) {
    freopen("1854.in", "r", stdin);
    freopen("1854.out", "w", stdout);
    scanf("%d%d%d", &N, &L, &P);
    for (int i = N - 1; i >= 0; i--) scanf("%d%d", &stop[i].dis, &stop[i].amount); 
    for (int i = 0; i < N; i++) stop[i].dis = L - stop[i].dis;
    sort(stop, stop + N);
    stop[N].dis = L; stop[N].amount = 0; //为了写起来方便,我们把终点也认为是加油站
    printf("%d\n", solve());
    return 0;
}


你可能感兴趣的:(解题报告,SMOJ,USACO,POJ,贪心)