会场安排问题
时间限制:3000 ms | 内存限制:65535 KB
难度:4
- 描述
- 学校的小礼堂每天都会有许多活动,有时间这些活动的计划时间会发生冲突,需要选择出一些活动进行举办。小刘的工作就是安排学校小礼堂的活动,每个时间最多安排一个活动。现在小刘有一些活动计划的时间表,他想尽可能的安排更多的活动,请问他该如何安排。
- 输入
- 第一行是一个整型数m(m<100)表示共有m组测试数据。
每组测试数据的第一行是一个整数n(1<n<10000)表示该测试数据共有n个活动。
随后的n行,每行有两个正整数Bi,Ei(0<=Bi,Ei<10000),分别表示第i个活动的起始与结束时间(Bi<=Ei)
输出 - 对于每一组输入,输出最多能够安排的活动数量。
每组的输出占一行 样例输入 -
2 2 1 10 10 11 3 1 10 10 11 11 20
样例输出 -
1 2
提示
注意:如果上一个活动在t时间结束,下一个活动最早应该在t+1时间开始
思想:这里运用贪心算法,先把结束的时间进行排序,再比较开始的时间。
qsort和sort两种排序方法,定义结构体,输入开始时间和结束时间。
#include<stdio.h> #include<stdlib.h> #define N 10001 struct node {int beg;int end; }table[N]; int cmp(const void* a,const void* b)//对结束时间进行排序 {return (*(struct node *)a).end-(*(struct node *)b).end; } int main() {int m;scanf("%d",&m);while(m--){int n,i,j;scanf("%d",&n);for(i=0;i<n;i++)scanf("%d%d",&table[i].beg,&table[i].end);//输入活动开始和结束时间qsort(table,n,sizeof(table[0]),cmp);int min=-1,count=0;for(i=0;i<n;i++){if(table[i].beg>min){count++;min=table[i].end;}} printf("%d\n",count);}return 0; }
此题主要利用时间安排表的贪心性质,可以先对结束时间进行排序,然后把第一个活动加入安排,因为如果第一个活动不加进去的话,后面的结束时间总是大于第一个活动的结束时间,所以满足不了局部最优,故加入第一个;之后要做的就是比较开始时间和结束时间,设起始时间Si和一个结束时间Fi,且Si<Fi,设活动i和j,且初始化i=1,j=0,只要有Si>Fj,就把活动i加入安排,并使j=i,然后i递增,直到i=n,即可统计活动安排个数;其中n数据在[0,10000),比较大,所以要使用快速排序,时间复杂度为nlog(n);具体代码如下:
#include<iostream> #include<stdio.h> #include<algorithm> using namespace std; #define N 10001 struct A {int beg;int end; }table[N]; int cmp(A a,A b) {return a.end<b.end?1:0; } void Input(int m) {int n,i,j;while(m--){scanf("%d",&n);for(i=0;i<n;i++)scanf("%d%d",&table[i].beg,&table[i].end);//输入活动开始和结束时间sort(table,table+n,cmp);int num=1;j=0;for(i=1;i<n;i++){if(table[i].beg>table[j].end){num++;j = i;}}printf("%d\n",num);} } int main() {int m;scanf("%d",&m);Input(m);return 0; }
- 第一行是一个整型数m(m<100)表示共有m组测试数据。