优化转移DP

Problem - H - Codeforces

题意

Aloha 要骑单车,可以单独花费 \(r\) 元骑 1 次,也可以购买某一种单车卡,第 \(i\) 种单车卡 \(c_i\) 元,若在第 \(t\) 天购买,可以在 \([t,t+d_i-1]\) 天使用,并且最多使用 \(k_i\) 次

给出 Aloha 一段时间内的单车使用情况,求出他需要的最小花费

给出

  1. \(n\;(1<=n<=500)\), 单车卡的种类

  2. \(m\;(1<=m<=10^5)\), \(m\) 条使用记录

  3. \(r\;(1<=r<=10^9)\), 单独骑 1 次的花费

  4. \(n\) 行,每行有 \(1<=d_i,k_i,c_i<=10^9\), 第 \(i\) 种单车卡的有效期、使用次数、价格

  5. \(m\) 行,每行有 \(0<=p_i<=10^9,\;0<=q_i<=3e5,\;\sum\limits_{i=1}^mq_i<=3e5\)

    表示在第 \(p_i\) 天骑了 \(q_i\) 次

思路

  1. 看上去就是 DP,观察数据范围,很多数据的值域过大,只有 \(n\) 和 \(\sum q_i\) 可能与 DP 复杂度有关

  2. 求出每次骑的时间 \(a[i]\), 一共骑了 cnt 次,并按时间排序;

    设 \(f[i]\) 为骑完前 \(i\) 次的最小花费,\(ptr[i][j]\) 为如果第 \(i\) 次可以用到第 \(j\) 个卡,则最早在第 \(ptr[i][j]\) 次骑完后买卡

    可列出朴素DP转移

    1. for (int i = 1; i <= cnt; i++)
    2. {
    3. f[i] = f[i - 1] + r;//初始化为不买卡直接骑
    4. for (int j = 1; j <= n; j++)
    5. {
    6. auto [d, k, c] = b[j];
    7. for (int w = ptr[i][j]; w < i; w++)
    8. f[i] = min(f[i], f[w] + c);
    9. }
    10. }
  3. 由于 \(f[i]\) 是单调不减的,因此 \(f[i]=min(f[i],f[w]+c)\) 中 \(w=ptr[i][j]\) 时是最优的(也可以感性地感受一下,这样卡的利用率更高,这样好像倒推更好理解一点,有时间试试)

  4. 接下来就是需要求出 \(ptr[i][j]\),第一维可以优化掉,\(ptr[j]\) 表示第 i 次骑车时,最早需要 \(ptr[j]\)次买第 j 种卡才能覆盖第 i 次

    随着 \(i\) 增大,\(ptr[j]\) 不会回退,因此可以暴力 check,双指针维护

代码

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. #define endl "\n"
  4. typedef long long ll;
  5. typedef pair<int, int> PII;
  6. const int N = 3e5 + 10;
  7. int n, m;
  8. ll r;
  9. struct Cards
  10. {
  11. ll d, k, c;
  12. }b[510];
  13. int a[N];//第i次的时间
  14. int cnt;
  15. ll f[N];//前i次的最小代价
  16. int ptr[510];
  17. bool check(int now, int card)
  18. {
  19. int last = ptr[card];
  20. auto [d, k, c] = b[card];
  21. if (last == now)
  22. return true;
  23. if (a[last] + d - 1 < a[now])
  24. return false;
  25. if (now - last + 1 > k)
  26. return false;
  27. return true;
  28. }
  29. int main()
  30. {
  31. ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
  32. cin >> n >> m >> r;
  33. for (int i = 1; i <= n; i++)
  34. cin >> b[i].d >> b[i].k >> b[i].c;
  35. for (int i = 1; i <= m; i++)
  36. {
  37. int p, q;
  38. cin >> p >> q;
  39. while(q--)
  40. a[++cnt] = p;
  41. }
  42. sort(a + 1, a + cnt + 1);
  43. for (int i = 1; i <= n; i++)
  44. ptr[i] = 1;
  45. for (int i = 1; i <= cnt; i++)
  46. {
  47. f[i] = f[i-1] + r;
  48. for (int j = 1; j <= n; j++)
  49. {
  50. auto [d, k, c] = b[j];
  51. // for (int w = ptr; w < i; w++)
  52. // f[i] = min(f[i], f[w] + c);
  53. // 因为f[i]单调递增,因此取w=x最小,ptr为在第ptr次后买优惠券,正好可以覆盖到第i次,双指针更新ptr
  54. while(!check(i, j))
  55. ptr[j]++;
  56. f[i] = min(f[i], f[ptr[j] - 1] + c);
  57. // cout << i << ": " << j << " " << ptr[j] << endl;
  58. }
  59. }
  60. cout << f[cnt] << endl;
  61. return 0;
  62. }

2020icpc沈阳H的更多相关文章

  1. 2020ICPC沈阳站C题 Mean Streets of Gadgetzan

    大致题意 原题链接 翻译 \(有n个逻辑变量 请你分别对它们赋值 使其满足m个命题\) \(命题有四种格式:\) 单独数字x 表示第x个逻辑变量为真 ! + 数字x 表示第x个逻辑变量为假 若干个数字 ...

  2. 每日一刷(2018多校水题+2016icpc水题)

    11.9 线段树 http://acm.hdu.edu.cn/showproblem.php?pid=6315 求逆序对个数 http://acm.hdu.edu.cn/showproblem.php ...

  3. 2019~2020icpc亚洲区域赛徐州站H. Yuuki and a problem

    2019~2020icpc亚洲区域赛徐州站H. Yuuki and a problem 题意: 给定一个长度为\(n\)的序列,有两种操作: 1:单点修改. 2:查询区间\([L,R]\)范围内所有子 ...

  4. 2016ACM/ICPC亚洲区沈阳站 - A/B/C/E/G/H/I - (Undone)

    链接:传送门 A - Thickest Burger - [签到水题] ACM ICPC is launching a thick burger. The thickness (or the heig ...

  5. 2016ACM/ICPC亚洲区沈阳站H - Guessing the Dice Roll HDU - 5955 ac自动机+概率dp+高斯消元

    http://acm.hdu.edu.cn/showproblem.php?pid=5955 题意:给你长度为l的n组数,每个数1-6,每次扔色子,问你每个串第一次被匹配的概率是多少 题解:先建成ac ...

  6. 2019沈阳icpc网络赛H德州扑克

    题面:https://nanti.jisuanke.com/t/41408 题意:A,2,3,4,5,6,7,8,9,10,J,Q,K,13张牌,无花色之分,val为1~13. 给n个人名+n个牌,输 ...

  7. 2016ACM/ICPC亚洲区沈阳站-重现赛赛题

    今天做的沈阳站重现赛,自己还是太水,只做出两道签到题,另外两道看懂题意了,但是也没能做出来. 1. Thickest Burger Time Limit: 2000/1000 MS (Java/Oth ...

  8. HDU 5950 Recursive sequence 【递推+矩阵快速幂】 (2016ACM/ICPC亚洲区沈阳站)

    Recursive sequence Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Other ...

  9. HDU 5952 Counting Cliques 【DFS+剪枝】 (2016ACM/ICPC亚洲区沈阳站)

    Counting Cliques Time Limit: 8000/4000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) ...

  10. HDU 5948 Thickest Burger 【模拟】 (2016ACM/ICPC亚洲区沈阳站)

    Thickest Burger Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)T ...

随机推荐

  1. 运维、监控、AIOps的几个重要观点

    监控是整个运维乃至整个产品生命周期中最重要的一环,通过配置合理的告警机制,采集准确的监控指标,来提前或者尽早发现问题,解决问题,进而保证产品的稳定,提升用户的体验.『分布式实验室』特约记者艾尔斯兰(下 ...

  2. Django AttributeError: 'BugDeserializer' object has no attribute '_meta'

    BugDeserializer 对象中没有 '_meta' 属性,定位到调用BugDeserializer位置, 用于序列化时,将模型类对象传入instance参数 在update时,数据传入有误,更 ...

  3. eclipse打不开maven项目中的pom.xml

    1.问题描述 在eclipse中我双击打开maven项目中的pom.xml报错. 2.产生原因 正如报错中描述的那样"Failed to create the part's controls ...

  4. linux开机自启动设置

    方法一:修改 /etc/rc.d/rc.local 文件/etc/rc.d/rc.local 文件会在 Linux 系统各项服务都启动完毕之后再被运行.所以想要脚本在开机后被运行的话,可以将自己脚本路 ...

  5. 为什么 java 容器推荐使用 ExitOnOutOfMemoryError 而非 HeapDumpOnOutOfMemoryError ?

    前言 好久没写文章了, 今天之所以突然心血来潮, 是因为昨天出现了这样一个情况: 我们公司的某个手机APP后端的用户(customer)微服务出现内存泄露, 导致OutOfMemoryError, 但 ...

  6. DVWA靶场——靶场搭建

    DVWA靶场搭建 一.phpstudy环境准备: 第一步:下载PHPstudy: 找到PHPstudy官网,下载windows版小皮面板,然后安装.官网下载地址:https://www.xp.cn/ ...

  7. AtCoder Beginner Contest 282 G - Similar Permutation

    套路题 题意 求有多少个 \(1\) 到 \(n\) 的排列满足恰有 \(k\) 对在排列中相邻的数满足前小于后 \(2 \leq n \leq 500, 0 \leq k \leq (n - 1)\ ...

  8. KMP 与 Z 函数

    \(\text{By DaiRuiChen007}\) 一.KMP 算法 I. 问题描述 在文本串 \(S\) 中找到模式串 \(T\) 的所有出现,其中 \(|S|=n,|T|=m\) II. 前置 ...

  9. Apache RocketMQ 5.0 笔记

    RocketMQ 5.0:云原生"消息.事件.流"实时数据处理平台,覆盖云边端一体化数据处理场景. 核心特性 云原生:生与云,长与云,无限弹性扩缩,K8s友好 高吞吐:万亿级吞吐保 ...

  10. Hugging Face 开源库介绍

    Hugging Face 的开源生态今年成长迅速,timm 成为新加入的成员.diffusers.evaluate 以及 skops 等各种库蓬勃发展. Transformers Transforme ...