P11290 【MX-S6-T2】「KDOI-11」飞船 题解

注意到速度的形式是编号相乘,而又有 \(x \in \{1, 2, 3, 4\}\),所以最多 \(\log_2y_i\) 次速度就会达到 \(10^9\) 量级,而此时再加油最少需要 \(1\) 秒,所以再乘一定不优。

直接 dp,有 \(f_{i, j, k}\) 表示当前在第 \(i\) 个加油站,速度为 \(2^j3^k\) 的最短用时,后面的 \(2^j3^k\) 可以递推处理。

具体的,转移考虑当前加油站是否加油即可。

对于询问,二分出当前位置前面的第一个加油站,然后枚举终点状态计算答案即可。

复杂度 \(O(n\log_2y\log_3y + q\log_2 y\log_3y)\)

// 如果命运对你缄默, 那就活给他看。
#pragma GCC optimize(1)
#pragma GCC optimize(2)
#pragma GCC optimize(3)
#pragma GCC optimize("Ofast", "inline", "-ffast-math")
#pragma GCC target("avx,sse2,sse3,sse4,mmx")
#include <cfloat>
#include <iostream>
#include <cstring>
#include <tuple>
#include <iomanip>
#include <cassert>
using namespace std;
typedef long long LL;
// typedef long double LD;
typedef double LD; 
// #define int LL
const int maxn = 100010;
inline LD cmi(LD& x, LD y) { if(y < x) x = y; return x; }
struct Port {
  int p, t, x;
  Port () {}
  Port (int a, int b, int c) { p = a, t = b, x = c; }
} w[maxn];
int n, q;
LL ve[31][20];
LD f[maxn][31][20];
inline int Bound(int y) {
  int l = 0, r = n;
  while(l < r) {
    int mid = l + r + 1 >> 1;
    if(w[mid].p <= y) l = mid;
    else r = mid - 1;
  } return l;
}
signed main() {
  // freopen("ship2.in", "r", stdin);
  // freopen("ship.out", "w", stdout);
  ios :: sync_with_stdio(false);
  cin.tie(0), cout.tie(0);
  cin >> n >> q;
  for(int i = 1; i <= n; ++ i) {
    int p, t, x;
    cin >> p >> t >> x;
    w[i] = {p, t, x};
  }
  ve[0][0] = 1;
  for(int j = 1; j < 20; ++ j) ve[0][j] = ve[0][j - 1] * 3; 
  for(int i = 1; i <= 30; ++ i) {
    ve[i][0] = ve[i - 1][0] * 2;
    for(int j = 1; j < 20; ++ j) {
      ve[i][j] = ve[i][j - 1] * 3;
    }
  }
  for(int i = 0; i <= n; ++ i) {
    for(int j = 0; j <= 30; ++ j) {
      for(int k = 0; k < 20; ++ k) {
        f[i][j][k] = 1e18;
      }
    }
  }
  f[0][0][0] = 0;
  for(int i = 1; i <= n; ++ i) {
    auto [p, t, x] = w[i];
    for(int j = 0; j <= 30; ++ j) {
      for(int k = 0; k < 20; ++ k) {
        cmi(f[i][j][k], f[i - 1][j][k] + 1.0 * (w[i].p - w[i - 1].p) / ve[j][k]);
        int j2 = j, k2 = k;
        if(x == 2) j2 -- ;
        else if(x == 3) k2 -- ;
        else if(x == 4) j2 -= 2;
        if(j2 < 0 || k2 < 0) continue ;
        LL v = ve[j2][k2];
        LD nt = 1.0 * (w[i].p - w[i - 1].p) / v;
        cmi(f[i][j][k], f[i - 1][j2][k2] + t + nt);
      }
    }
  }
  for(int y; q -- ; ) {
    cin >> y;
    LD ans = DBL_MAX;
    int x = Bound(y);
    for(int i = 0; i <= 30; ++ i) {
      for(int j = 0; j < 20; ++ j) {
        LL v = ve[i][j];
        LD nt = 1.0 * (y - w[x].p) / v;
        cmi(ans, f[x][i][j] + nt);
      }
    }
    cout << fixed << setprecision(7) << ans << '\n';
  }
  return 0;
}

调的时候挂在了奇妙的地方:memset 一个 long double 数组 \(0x7f\) 会出来 nan /hsh。

posted @ 2024-11-20 11:48  Rainsheep  阅读(3)  评论(0编辑  收藏  举报