基于贪心算法的几类区间覆盖问题 nyoj 12喷水装置(二) nyoj 14会场安排问题

1)区间完全覆盖问题

问题描述:给定一个长度为m的区间,再给出n条线段的起点和终点(注意这里是闭区间),求最少使用多少条线段可以将整个区间完全覆盖

样例:

区间长度8,可选的覆盖线段[2,6],[1,4],[3,6],[3,7],[6,8],[2,4],[3,5]

解题过程:

1将每一个区间按照左端点递增顺序排列,拍完序后为[1,4][2,4][2,6][3,5][3,6][3,7][6,8]

2设置一个变量表示已经覆盖到的区域。再剩下的线段中找出所有左端点小于等于当前已经覆盖到的区域的右端点的线段中,右端点最大的线段在加入,直到已经覆盖全部的区域

3过程:

假设第一步加入[1,4],那么下一步能够选择的有[2,6][3,5][3,6][3,7],由于7最大,所以下一步选择[3,7],最后一步只能选择[6,8],这个时候刚好达到了8退出,所选区间为3

4贪心证明:

需要最少的线段进行覆盖,那么选取的线段必然要尽量长,而已经覆盖到的区域之前的地方已经无所谓了,(可以理解成所有的可以覆盖的左端点都是已经覆盖到的地方),那么真正能够使得线段更成的是右端点,左端点没有太大的意义,所以选择右端点来覆盖


NYOJ 12 http://acm.nyist.net/JudgeOnline/problem.php?pid=12

喷水装置(二)

 1 #include<stdio.h>
 2 #include<algorithm>
 3 #include<math.h>
 4 using namespace std;
 5 struct node
 6 {
 7     double begin,end;
 8 }r[10001];
 9 bool cmp(node x,node y)
10 {
11     return x.begin<y.begin;
12 }
13 int main()
14 {
15     int t,n,w,h,i,j;
16     double x,ri,len;
17     scanf("%d",&t);
18     while(t--)
19     {
20         scanf("%d %d %d",&n,&w,&h);
21         for(i=0;i<n;i++)
22         {
23             scanf("%lf %lf",&x,&ri);
24             len=4*ri*ri-h*h;
25             if(len<0)
26             len=0;
27             else
28             len=0.5*sqrt(len);
29             r[i].begin=x-len;
30             r[i].end=x+len;
31         }
32         int ans=0,f;
33         sort(r,r+n,cmp);
34         double max=0,sum=0;
35         f=1;
36         i=0;
37         while(sum<w)
38         {
39             max=0;
40             for(int k=0;k<n;k++)//找最大的右端点 
41             {
42                 if(r[k].begin<=sum&&max<r[k].end-sum)
43                 {
44                     max=r[k].end-sum;
45                 }
46             }
47             if(max==0)
48             {
49                 f=0;
50                 break;
51             }
52             ans++;
53             sum+=max;
54         }
55         if(f)
56         printf("%d
",ans);        
57         else
58         printf("0
");
59     }
60     
61     return 0;
62 }
View Code

2)最大不相交覆盖

问题描述:给定一个长度为m的区间,再给出n条线段的起点和终点(开区间和闭区间处理的方法是不同,这里以开区间为例),问题是从中选取尽量多的线段,使得每个线段都是独立的,就是不和其它有任何线段有相交的地方


样例:

区间长度8,可选的覆盖线段[2,6],[1,4],[3,6],[3,7],[6,8],[2,4],[3,5]

解题过程:

对线段的右端点进行升序排序,每加入一个线段,然后选择后面若干个(也有可能是一个)右端点相同的线段,选择左端点最大的那一条,如果加入以后不会跟之前的线段产生公共部分,那么就加入,否则就继续判断后面的线段

1排序:将每一个区间按右端点进行递增顺序排列,拍完序后为[1,4][2,4][2,6][3,5][3,6][3,7][6,8]

2第一步选取[2,4],发现后面只能加入[6,8],所以区间的个数为2


贪心证明:因为需要尽量多的独立的线段,所以每个线段都尽可能的小,对于同一右端点,左端点越大,线段长度越小。那么为什么要对右端点进行排序呢?如果左端点进行排序,那么右端点是多少并不知道,那么每一条线段都不能对之前所有的线段进行一个总结,那么这就明显不满足贪心的最有字结构了。

 http://acm.nyist.net/JudgeOnline/problem.php?pid=14    NYOJ 14

 1  
 2 #include<stdio.h>
 3 #include<algorithm>
 4 using namespace std;
 5 struct node
 6 {
 7     int end;
 8     int begin;
 9 }a[10001];
10 bool cmp(node x,node y)
11 {
12     return x.end<y.end;
13 }
14 int main()
15 {
16     int t;
17     int n,i,j;
18     scanf("%d",&t);
19     while(t--)
20     {
21         int ans=1;
22         scanf("%d",&n);
23         for(i=0;i<n;i++)
24         {
25             scanf("%d %d",&a[i].begin,&a[i].end);
26         }
27         sort(a,a+n,cmp);
28         j=0;
29         for(i=1;i<n;i++)
30         {
31             if(a[j].end<a[i].begin)
32             {
33                 ans++;
34                 j=i;
35             }
36         }
37         printf("%d
",ans);
38     }
39     return 0;
40     
41 }        
View Code



3)区间选点问题

问题描述:给定一个长度为m的区间,再给出n条线段和这n条线段需要满足的要求(要求是这n条线段上至少有的被选择的点的个数),问题是整个区间内最少选择几个点,使其满足每一条线段的要求.

样例:略

解题过程:将每个线段按照终点坐标进行递增排序,相同终点的前点坐标大的在前面,一个个将其满足

贪心证明:要想使得剩下的线段上选择的点最少,那么就应该尽量使得已经选择了的点尽量能在后面的线段中发挥作用,而我们是从左往右选择线段的,那么要使得选取的点能满足后面线段的要求,那么必须是从线段的有端点开始选点,那么问题(2)一样涉及到一个问题,如果是按照线段的左端点对线段进行排序的话,不知道右端点的话,每一条线段都不能对之前已经操作过的所有线段进行一个总结,那么这就同样不满足贪心算法的最优子结构性质了。

可以解决的实际问题:数轴上面有n个闭区间[a,b],取尽量少的点,使得每个区间内都至少有一个点(不同区间内含的点可以是同一个)

原文地址:https://www.cnblogs.com/zeze/p/summer1214.html