zoukankan      html  css  js  c++  java
  • HDU 6249 Alice’s Stamps

    题目链接

    题目大意:

    说有$m$个区间,要求选出不超过$k$个区间,使这些区间覆盖的长度最长,问最长长度是多少。

    题解:

    所有区间按$R$从小到大排序之后可以进行$dp$。

    $dp[i][j]$表示:拿了小于等于$i$个区间,最后一个以坐标小于等于$j$为结尾的最长覆盖长度

    假设第$x$个区间作为结尾,那么要分两种情况来考虑:

    1.可以是之前的结尾小于第$x$个区间的左端点,这种情况很好解决。

    2.也可以是之前区间的结尾在第$x$个区间内部。

    第二种情况的话:

    不允许在区间内部进行枚举点,否则时间复杂度炸了,可以发现要求的是类似于$dp[j] + i - j$格式的最大值,也就是$i$加上区间上$dp[j]-j$的最大值,因此可以用ST表计算区间最大值。

    #include <bits/stdc++.h>
    using namespace std;
    
    const int maxn = 2100;
    int T, n, m, K;
    struct X {
      int L, R;
      int x;
    }s[maxn];
    int dp[maxn][maxn];
    int t[maxn * 4];
    int rmq[maxn][15];
    
    bool cmp(const X&a, const X&b) {
      return a.R < b.R;
    }
    
    void ST(int num) {
      for (int i = 1; i <= n; i++)
        rmq[i][0] = dp[num][i] - i;
      for (int j = 1; (1 << j) <= n; j++) {
        for (int i = 1; i + (1 << j) - 1 <= n; i++) {
          rmq[i][j] = max(rmq[i][j - 1], rmq[i + (1 << (j - 1))][j - 1]);
        }
      }
    }
    
    int RMQ(int l, int r) {
      int k = 0;
      while ((1 << (k + 1)) <= r - l + 1) k++;
      return max(rmq[l][k], rmq[r - (1 << k) + 1][k]);
    }
    
    int main() {
      scanf("%d", &T);
      int cas = 1;
      while(T --) {
        scanf("%d%d%d", &n, &m, &K);
        for(int i = 1; i <= m; i ++) {
          scanf("%d%d", &s[i].L, &s[i].R);
          s[i].x = s[i].R - s[i].L + 1;
        }
        sort(s + 1, s + 1 + m, cmp);
        /*
         dp[i][j], 拿了 <= i 个,最后一个以坐标 <= j 为结尾
         */
        int ans = 0;
        ST(0);
        for(int i = 1; i <= K; i ++) {
          int now = 1;
          while(now <= m && s[now].R < i) now ++;
          for(int j = i; j <= n; j ++) {
            dp[i][j] = 0;
            while(now <= m && s[now].R == j) {
              dp[i][j] = max(dp[i][j],
                                s[now].x + dp[i - 1][s[now].L - 1]);
              dp[i][j] = max(dp[i][j],
                                RMQ(s[now].L, s[now].R) + j);
              now ++;
            }
            dp[i][j] = max(dp[i][j - 1], dp[i][j]);
            ans = max(ans, dp[i][j]);
            if(ans == n) break;
          }
          if(ans == n) break;
          ST(i);
        }
        printf("Case #%d: %d
    ", cas ++, ans);
      }
      return 0;
    }
    
  • 相关阅读:
    结对第一次—疫情统计可视化(原型设计)
    软工实践寒假作业(2/2)
    test
    软工实践寒假作业(1/2)
    json的基本用法
    个人作业——软件工程实践总结&个人技术博客
    个人作业——软件评测
    结对第二次作业——某次疫情统计可视化的实现
    寒假作业(2/2)— 疫情统计
    软工实践寒假作业(1/2)
  • 原文地址:https://www.cnblogs.com/zufezzt/p/8430868.html
Copyright © 2011-2022 走看看