HITSZ数据结构实验作业
我的做题经历,可以给大家一点点参考
问题
首先输入两个数字N,M,表示N个空闲时间段,M个学生。接着程序输入M行,每行两个数字,表示第i名同学空闲时间的开始时间段与终止时间段。其中1≤N≤100000000 , 1≤M≤1000。
程序输出成对的数字,表示空闲人数最多时间段的起始时间和终止时间,如若有多对,请以递增形式在一行中输出。不同对之间用英文逗号“,”分隔,对内元素用空格隔开。
当两个时间段的空闲学生不完全一致时,我们认为其为不同时间段。如若完全一致,则为相同时间段。
要求:排序算法的平均时间复杂度不得大于O(Mlog2M)。
分析
直接使用长度为n的线性表表示各个时间点是最容易想到的办法,但是这样需要的空间非常非常大,并且时间复杂度也远大于要求.
进一步分析,如果按照上述建立存储结构,那么数组中间有很多部分其实是相等的,人数真正发生变化的时间点其实就是我们每次输入的同学起始时间和终止时间.我们可以认为在后m行输入的这些时间点是关键时间点,我们只需要分析这些时间点即可.
1.建立两个数组IN,OUT
IN数组表示各个起始时间
OUT表示各个终止时间
2.建立时间点结构体Time
typedef struct Time
{
int mark;//时间点的位置
int pop;//该点的人数
} Time;
注意:终止时间点的下一个时间点才是人数减少的时间点
第一步:输入
int n, m; // n:接受时间段,m接受学生数量(其实n没什么用)
//输入
scanf("%d", &n);
scanf("%d", &m);
int counter = 0;
for (counter = 0; counter < m; counter++)
scanf("%d %d", &In[counter], &Out[counter]);
第二步:排序
使用任意方法将IN数组和OUT数组进行排序即可
如果不排序,那么我们寻找关键时间点并进行操作时则必须要建立一个一一对应的key.
第三步:将IN和OUT数组归并
归并这两个数组的原因:得到一个Schedule数组,这个数组存储的信息就是所有人数发生变化的时间点,之后我们只需要按照顺序,将Schedule表与IN以及OUT分别比较即可.
第四步:得到所有关键点的人数
timelist : Time结构体数组,存储的是关键时间点
将schedule数组里重复的元素删除得到changetime数组,可以把changetime当作一串指令,指令内容如下
- 如果changtime[n]与IN [i] 相同,那么将IN[i]记录为timelist[counter++].mark,人数timelist[counter-1].pop+1(如果有重复那么则自加);
- 如果changetime[n]与OUT[j]相同,那么OUT[j]记录为timelist[counter+1].mark,人数为timelist[counter].pop-1(如果有重复那么则自减);
第五步:找到人数最多的关键点
略过
第六步:输出
第五步我们已经找出所有人数最多的关键点
那么输出就非常简单了.
人数最多的关键点到下一个关键点的前一个点就是人数最多的时间段,直接输出即可.
一些我遇到过的问题
-
当两个时间段的空闲学生不完全一致时,我们认为其为不同时间段。如若完全一致,则为相同时间段。这个要求如何实现?我们仔细分析就可以发现,因为每次输入IN,OUT时都是不同的同学,那么说明不同关键点中包含的学生绝对不可能相同
举个例子:
假设
pop mark timelist[2].pop=10 (max) timelist[2].mark=2 timelist[3].pop=10 (max) timelist[3].mark=5 timelist[4].pop=4 timelist[4].mark=6 timelist[2],timelist[3]这两个点都被选作关键点,说明在timelist[2]之后走了一部分人,在timelist[3]中又进来了相等数量的其他人,所以可以认为是不同的时间段.输出时就是
2 4, 5 5
完整代码:
#include <stdio.h>
#include <stdlib.h>
#define MAX_STUDENT 1000
typedef struct Time
{
int mark;
int pop;
} Time;
/**
* @brief 对两个等长数组进行归并
*
* @param a 数组a
*

最低0.47元/天 解锁文章
1176





