51Nod:活动安排问题之二(贪心)

有若干个活动,第i个开始时间和结束时间是[Si,fi),同一个教室安排的活动之间不能交叠,求要安排所有活动,最少需要几个室? 

输入
第一行一个正整数n (n <= 10000)代表活动的个数。
第二行到第(n + 1)行包含n个开始时间和结束时间。
开始时间严格小于结束时间,并且时间都是非负整数,小于1000000000
输出
一行包含一个整数表示最少教室的个数。
输入示例
3
1 2
3 4
2 9
输出示例
2

有若干个活动,第i个开始时间和结束时间是[Si,fi),活动之间不能交叠,要把活动都安排完,至少需要几个教室?
分析:能否按照之一问题的解法,每个教室安排尽可能多的活动,即按结束时间排序,再贪心选择不冲突的活动,安排一个教室之后,剩余的活动再分配一个教室,继续贪心选择……

反例: A:[1,2)  B:[1,4) C:[5,6) D:[3,7)
已经按结束时间排好顺序,我们会选择
教室1: A C
教室2:  B
教室3:  D
需要3个教室

但是如果换一种安排方法,我们可以安排AD在一个教室,而BC在另外一个教室,两个教室就够了。
所以之前的贪心策略解决不了这个问题。
怎么办?之前的策略是用一个教室找所有它能安排下的活动,即用教室找活动,我们能不能用活动找教室呢?

策略: 按照开始时间排序优先安排活动,如果冲突,则加一个教室。
简单地理解一下,策略是这样,我们把活动按照开始时间有小到大的顺序排序。假设目前已经分配了k个教室(显然k初始等于0),对于当前这个活动,
(1) 如果它能安排在k个教室里的某一个,则把它安排在其中的任何一个教室里,k不变。
(2) 否则它和每个教室里的活动都冲突,则增加一个教室,安排这个活动。

这个策略是最优么?

我们想像一下k增加1的过程: 因为我们是按照开始时间排序的,意味着当前考虑的这个活动开始的时候,k个教室里都有活动没结束(因为如果有一个教室的活动结束了,我们就可以安排这个活动进入那个教室而不冲突,从而不用增加k)。这就意味着在这个活动开始的时间点,算上目前考虑的这个活动,有(k + 1)个活动正在进行,同一时刻有(k + 1)个活动在进行,无论我们如何安排教室,都至少需要(k + 1)个教室。因为每个教室里不能同时进行两个活动。而我们的策略恰好需要(k + 1)个教室,所以是最优的。

这个策略也告诉我们,如果从时间轴上“宏观”考虑这个问题。考虑每个时间点同时进行的活动个数,作为这个时间点的厚度(把活动开始和结束时间想像成线段,那么每个时间点有多少条线段覆盖它,可以简单理解为“厚度”),我们至少需要最大厚度那么多个教室——因为那时恰好有最大厚度那么多个活动同时进行,而我们这个贪心策略恰好给了我们一个用最大厚度那么多个教室安排全部活动的一个方案。

如果只需要教室的个数,我们可以把所有开始时间和结束时间排序,遇到开始时间就把厚度加1,遇到结束时间就把厚度减1,显然最初始和最后结束时的厚度是0,在一系列厚度变化的过程中,峰值(最大值)就是最多同时进行的活动数,也是我们至少需要的教室数。

方法一:就是51Nod里面讲解的方法(上面的绿色部分),不需要使用结构体

AC代码:

#include<bits/stdc++.h>
#define x 10010
using namespace std;
int main()
{
	int s[x],e[x];
	int n,i,k;
	scanf("%d",&n);
	for(i=0;i<n;i++) scanf("%d%d",&s[i],&e[i]);
	sort(s,s+n);
	sort(e,e+n);
	int sum=0;
	for(i=0,k=0;i<n;i++)
	{
		if(s[i]<e[k]) sum++;
		else k++;
	}
	printf("%d
",sum);
	return 0;
} 

方法二:使用结构体(这种方法思路和方法一基本上一样,但是时间复杂度大了好多)

//还有别的复杂度小的方法,但是水平有限,百度到的方法不(kan)会(bu)用(dong)

AC代码:

#include<bits/stdc++.h>
#define x 10100
using namespace std;
struct wzy{
	int start,end;
}p[x];
int cmp(wzy u,wzy v)//对时间进行排序,
{
	if(u.start!=v.start) return u.start<v.start;//开始时间不同,把先开始的放前面 
	else return u.end<v.end;//开始时间相同,把先结束的放前面 
}
int main()
{
	int n,i,j,k,sum;
	scanf("%d",&n);
	for(i=0;i<n;i++) scanf("%d%d",&p[i].start,&p[i].end);
	sort(p,p+n,cmp);
    for(sum=0,i=0;i<n;i++)
    {
        k=0;
        for(j=i;j>=0;j--)//每次都要从i往前推一边,找出来在第i组数据的时候需要几个房间 
        {				//这个思路和方法一大致相同,都是找结束时间与开始时间的关系。从j=i开始是因为无论怎样都至少需要一个房间 
            if(p[j].end>p[i].start)
                k++;
        }
        sum=sum>k?sum:k;//取最大值 
    }
	printf("%d
",sum);
	return 0;
}
原文地址:https://www.cnblogs.com/Friends-A/p/9309057.html