Time Limit: 1000MS   Memory Limit: 65536K
Total Submissions: 4721   Accepted: 1593

Description

Farmer John's cows, pampered since birth, have reached new heights of fastidiousness. They now require their barn to be immaculate. Farmer John, the most obliging of farmers, has no choice but hire some of the cows to clean the barn.

Farmer John has N (1 <= N <= 10,000) cows who are willing to do some cleaning. Because dust falls continuously, the cows require that the farm be continuously cleaned during the workday, which runs from second number M to second number E during the day (0 <= M <= E <= 86,399). Note that the total number of seconds during which cleaning is to take place is E-M+1. During any given second M..E, at least one cow must be cleaning.

Each cow has submitted a job application indicating her willingness to work during a certain interval T1..T2 (where M <= T1 <= T2 <= E) for a certain salary of S (where 0 <= S <= 500,000). Note that a cow who indicated the interval 10..20 would work for 11 seconds, not 10. Farmer John must either accept or reject each individual application; he may NOT ask a cow to work only a fraction of the time it indicated and receive a corresponding fraction of the salary.

Find a schedule in which every second of the workday is covered by at least one cow and which minimizes the total salary that goes to the cows.

Input

Line 1: Three space-separated integers: N, M, and E.

Lines 2..N+1: Line i+1 describes cow i's schedule with three space-separated integers: T1, T2, and S.

Output

Line 1: a single integer that is either the minimum total salary to get the barn cleaned or else -1 if it is impossible to clean the barn.

Sample Input

  1. 3 0 4
  2. 0 2 3
  3. 3 4 2
  4. 0 0 1

Sample Output

  1. 5

Hint

Explanation of the sample:

FJ has three cows, and the barn needs to be cleaned from second 0 to second 4. The first cow is willing to work during seconds 0, 1, and 2 for a total salary of 3, etc.

Farmer John can hire the first two cows.

Source

题意就是给你一些小区间,用小区间去覆盖大区间,求最小花费。

数据结构优化动态规划,这是一个区间最值问题。

将小区间按照右端点排序,然后遍历小区间,每次都找小区间左边到小区间右边的最小值,然后最小值更新,只单点更新右端点就可以(降维)。

一开始想的区间更新,按照贴纸,直接成段更新,后来发现思路是错的,因为区间更新会覆盖掉其他的值。所以要用单点更新。

线段树每个点表示到当前区间段的最优解,直接初始化为inf,然后M-1更新为0,就可以。

每次都是查找a[i].l-1到a[i].r的最小值(因为是片段,所以从a[i].l-1开始查找),然后更新a[i].r。直接线段树维护区间的最小值。

动态规划方程式就是dp[i]=min(query(a[i].l-1,a[i].r))+a[i].val;表示覆盖到a[i].r的最小花费。因为小区间不能间断,所以要这样查询更新。

因为直接遍历的,所以dp[i]这个方程式都可以省略,直接查值然后更新值到线段树里就可以了。

因为每次都是从a[i].l-1开始,而且题目数据最小可能为0,所以直接端点都+2,这样,最小的值查找的时候也是从1开始的。

写题的时候,不能乱秀,容易把自己秀死。。。

自己写的时候,特判了一下如果最左端点和最端点不能包含大区间,直接-1输出。结果忘了我是按照右端点排序的,右边是对的,左边的不一定是对的。。。

最后直接查询大区间右端点和小区间最右端点。判断一下值就可以了。

如果ans<inf说明都能覆盖到,如果有inf说明有的地方没被覆盖到。

代码:

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. #include<algorithm>
  5. #include<bitset>
  6. #include<cassert>
  7. #include<cctype>
  8. #include<cmath>
  9. #include<cstdlib>
  10. #include<ctime>
  11. #include<deque>
  12. #include<iomanip>
  13. #include<list>
  14. #include<map>
  15. #include<queue>
  16. #include<set>
  17. #include<stack>
  18. #include<vector>
  19. using namespace std;
  20. typedef long long ll;
  21. typedef long double ld;
  22. typedef pair<int,int> pii;
  23.  
  24. const double PI=acos(-1.0);
  25. const double eps=1e-;
  26. const ll mod=1e9+;
  27. const ll inf=1e18;
  28. const int maxn=1e5+;
  29. const int maxm=+;
  30. #define ios ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
  31. #define lson l,m,rt<<1
  32. #define rson m+1,r,rt<<1|1
  33.  
  34. struct node{
  35. int l,r;
  36. ll val;
  37.  
  38. friend bool operator<(node a,node b){
  39. if(a.r==b.r) return a.l<b.l;
  40. return a.r<b.r;
  41. }
  42.  
  43. }a[maxn];
  44.  
  45. ll tree[maxn<<];
  46.  
  47. void pushup(int rt)
  48. {
  49. tree[rt]=min(tree[rt<<],tree[rt<<|]);
  50. }
  51.  
  52. void build(int l,int r,int rt)
  53. {
  54. if(l==r){
  55. tree[rt]=inf;
  56. return ;
  57. }
  58.  
  59. int m=(l+r)>>;
  60. build(lson);
  61. build(rson);
  62. pushup(rt);
  63. }
  64.  
  65. void update(int pos,ll c,int l,int r,int rt)
  66. {
  67. if(l==r){
  68. tree[rt]=min(tree[rt],c);
  69. return ;
  70. }
  71.  
  72. int m=(l+r)>>;
  73. if(pos<=m) update(pos,c,lson);
  74. if(pos> m) update(pos,c,rson);
  75. pushup(rt);
  76. }
  77.  
  78. ll query(int L,int R,int l,int r,int rt)
  79. {
  80. if(L<=l&&r<=R){
  81. return tree[rt];
  82. }
  83.  
  84. int m=(l+r)>>;
  85. ll ret=inf;
  86. if(L<=m) ret=min(ret,query(L,R,lson));
  87. if(R> m) ret=min(ret,query(L,R,rson));
  88. return ret;
  89. }
  90.  
  91. int main()
  92. {
  93. int n,M,E;
  94. cin>>n>>M>>E;
  95. for(int i=;i<=n;i++){
  96. cin>>a[i].l>>a[i].r>>a[i].val;
  97. a[i].l+=,a[i].r+=;
  98. }
  99. M+=,E+=;
  100. sort(a+,a++n);
  101. build(,maxn,);
  102. update(M-,,,maxn,);
  103. for(int i=;i<=n;i++){
  104. ll cnt=query(a[i].l-,a[i].r,,maxn,);
  105. cnt+=a[i].val;
  106. update(a[i].r,cnt,,maxn,);
  107. }
  108. ll ans=query(E,a[n].r,,maxn,);
  109. if(ans<inf) cout<<ans<<endl;
  110. else cout<<"-1"<<endl;
  111. }

OK了。

POJ 3171.Cleaning Shifts-区间覆盖最小花费-dp+线段树优化(单点更新、区间查询最值)的更多相关文章

  1. poj 3171 Cleaning Shifts(区间的最小覆盖价值)

    Cleaning Shifts Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 2743   Accepted: 955 De ...

  2. POJ 2376 Cleaning Shifts 区间覆盖问题

    http://poj.org/problem?id=2376 题目大意: 给你一些区间的起点和终点,让你用最小的区间覆盖一个大的区间. 思路: 贪心,按区间的起点找满足条件的并且终点尽量大的. 一开始 ...

  3. POJ 2892 Tunnel Warfare || HDU 1540(树状数组+二分 || 线段树的单点更新+区间查询)

    点我看题目 题意 :N个村子连成一条线,相邻的村子都有直接的地道进行相连,不相连的都由地道间接相连,三个命令,D x,表示x村庄被摧毁,R  ,表示最后被摧毁的村庄已经重建了,Q x表示,与x直接或间 ...

  4. POJ 3171 Cleaning Shifts(DP+zkw线段树)

    [题目链接] http://poj.org/problem?id=3171 [题目大意] 给出一些区间和他们的价值,求覆盖一整条线段的最小代价 [题解] 我们发现对区间右端点排序后有dp[r]=min ...

  5. POJ 3171 Cleaning Shifts

    Description Farmer John's cows, pampered since birth, have reached new heights of fastidiousness. Th ...

  6. [USACO2005][POJ3171]Cleaning Shifts(DP+线段树优化)

    题目:http://poj.org/problem?id=3171 题意:给你n个区间[a,b],每个区间都有一个费用c,要你用最小的费用覆盖区间[M,E] 分析:经典的区间覆盖问题,百度可以搜到这个 ...

  7. POJ - 2376 Cleaning Shifts 贪心(最小区间覆盖)

    Cleaning Shifts Farmer John is assigning some of his N (1 <= N <= 25,000) cows to do some clea ...

  8. poj3171 Cleaning Shifts【线段树(单点修改区间查询)】【DP】

    Cleaning Shifts Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 4422   Accepted: 1482 D ...

  9. HDU 1540 Tunnel Warfare 平衡树 / 线段树:单点更新,区间合并

    Tunnel Warfare                                  Time Limit: 4000/2000 MS (Java/Others)    Memory Lim ...

随机推荐

  1. windows版本redis下载安装

    官方网站:http://redis.io/ 官方下载:http://redis.io/download 可以根据需要下载不同版本 在官方下载网页中有一个window版本的说明,说redis官方没有wi ...

  2. 【51NOD-0】1049 最大子段和

    [算法]DP [题解]开long long…… #include<cstdio> #include<algorithm> #include<cstring> usi ...

  3. quick-cocos2d-x数据存储 UserDefault GameState io

    看了quick-cocos2d-x 的framework,发现里面有一个GameState,查了下,是数据存储的类,于是稍稍总结下我用到过的数据存储方式吧. 一共是三种方法: cc.UserDefau ...

  4. Spring Boot 中使用 MyBatis 整合 Druid 多数据源

    2017 年 10 月 20 日   Spring Boot 中使用 MyBatis 整合 Druid 多数据源 本文将讲述 spring boot + mybatis + druid 多数据源配置方 ...

  5. java map 转 json 自编封装

    1.自编封装代码: import com.alibaba.fastjson.JSON; import java.util.*; public class jsonConversion { privat ...

  6. CSS3 动画实现方法大全

    常用效果总结(需要引用animate.css) <!doctype html> <html lang="en"> <head> <meta ...

  7. highcharts 从后台动态改变数据

    //columnChart    图表对象,创建示例就展示了. var series = this.columnChart.series;                            whi ...

  8. arpspoof dnsspoof中间人攻击

    最近搞了一个监听神器,尽管使用了网卡混杂模式,不过监听到的几乎全是本地流量, 为了获取更多有用的数据,搞一下中间人攻击,最基本的就是arpspoof + IP转发,这样就可以获得局域网内任何人的上网流 ...

  9. 学习 Linux,101: 自定义或编写简单脚本【转】

    转自:http://www.ibm.com/developerworks/cn/linux/l-lpic1-105-2/index.html 学习如何使用标准的 shell 语法.循环和控制结构,以及 ...

  10. linux 系统调用exec()

    系统调用execve()对当前进程进行替换,替换者为一个指定的程序,其参数包括文件名(filename).参数列表(argv)以及环境变量(envp).exec函数族当然不止一个,但它们大致相同,在 ...