心血来潮做了老师布置的一道ACM题,思前想后,加之老师的引导才弄出结果来,不过做完之后发现一个老结论——问题都是在解决之前很困难,解决之后你会发现,喔~原来是这样....现在下笔记之,只为了提高一下自己的语言即逻辑组织能力,别无它用...
【1楼____题目】
【2楼____题意简析】 【3楼____题目流程逻辑化】 【4楼____数据结构分析】
【5楼____题目关键细节分析】
【6楼____程序实现思路】
【7楼____访问流程部分的源代码】
//------------------------------------------------1st floor--------------------------------------------------
【题目】
http://poj.org/problem?id=1025
//------------------------------------------------2nd floor--------------------------------------------------
【题意简析】
故事背景: 他们说有一栋总部大楼,每天有若干名神秘人员要进入总部去各个房间呆呆,然后,悄无声息地离开......好冷....进入正题 题意简析: 有一个10层的总部大楼,每层楼有10个房间以及1部电梯(这里强调下,区别于我们日常生活中常见的各类电梯,本总部的电梯比较理想,5秒一班,只要你欲乘上电梯的时间是5的倍数即可搭乘,另一特殊情况5楼详述 )。 每天有若干访问者要进入大楼对楼中某些房间进行访问并在房中停留一段时间,最后离开大楼。 每个访问者都有自己的唯一标识,由字母A~Z表示,字母越靠前,该访问者优先级越高。 每个房间及电梯在同一时间只能进一个人,如果有多名人员要访问同一个房间或电梯,他们必须在外排队, 排队的标准是:优先级越高的访问者越靠前。
每 个访问者都先进入大楼(即1楼) ,找接待员拿自己的访问清单(该清单记录了此人在不受任何影响情况下的初始访问流程)。 每个访问者都以清单中楼层以及房间号的升序遍历房间,如从0101开始到0206,如果欲访问的房间中有人,他就必须在外等待,而非跳过本房间去下一个房间。 在完成最后一个房间的访问后,访问者们离开大楼(如果不在1楼,访问者们要通过电梯降到 1 楼,然后离开大楼) 时间细节: 访问者进入总部大楼,得到接待员给的访问清单,到达电梯前或者到达第一层的指定房间,需要 30秒。 离开总部大楼,即从一楼的电梯出来或者从一楼的房间里出来经过接待员走出总部大楼需要 30秒。 同一层楼时,从电梯到指定房间或者指定房间前的排队队列;同一层楼时,从房间走到电梯里或者电梯前的排队队列;同一层楼时,从一个房间出来走到另一个房间或者另一个房间前的排队队列, 都需要 10秒钟。 使用电梯上、下一层,需要30秒。 输入数据:
A 10:00:00 //访问者标识及其进入大楼的开始时间
0101 100 //访问者欲访问的楼层房间号及欲在其中停留的时间
0110 50
0202 90
0205 50
0 //某一个访问者的数据结束标记B 10:00:000101 200
. //整个输入数据结束标记 输出数据: 按访问者优先级由高到低输出每个访问者详细的访问流程,第一排是访问者标识,以后每排输出一个流程,最后以空白行结束。 流程描述有以下语句:
Entry
Waiting in elevator queue
Waiting in front of room xxyy
Transfer from room xxxx to room xxyy
Transfer from elevator to room xxyy
Transfer from room xxyy to elevator
Stay in room xxyy
Stay in elevator
Exit 具体如: A 10:00:00 10:00:30 Entry 10:00:30 10:02:10 Stay in room 0101 10:02:10 10:02:20 Transfer from room 0101 to room 0110 1 0:02:20 10:03:10 Stay in room 0110 1 0:03:10 10:03:20 Transfer from room 0110 to elevator 1 0:03:20 10:03:50 Stay in elevator 1 0:03:50 10:04:00 Transfer from elevator to room 0202 1 0:04:00 10:05:30 Stay in room 0202 1 0:05:30 10:05:40 Transfer from room 0202 to room 0205 1 0:05:40 10:07:40 Waiting in front of room 0205 1 0:07:40 10:08:30 Stay in room 0205 1 0:08:30 10:08:40 Transfer from room 0205 to elevator 1 0:08:40 10:09:10 Stay in elevator 1 0:09:10 10:09:40 Exit 输入输出详见原题
//------------------------------------------------3rd floor--------------------------------------------------
【题目流程逻辑化】
现在将题目描述的访问流程提炼成逻辑流程。 通过对题意的解析,不难发现对于一个访问者来说,我们能够提炼出以下9个基本逻辑事件:
①进入大楼
②等待电梯
③等待进入房间
④从房间走到房间(指同楼层的房间)
⑤从电梯走到指定房间
⑥从房间走到电梯
⑦呆在房间中
⑧在电梯中
⑨离开大楼 对于上述分析,我们可以列出一个事件表格,并对其标以事件类型以便程序实现
事件描述 | 事件类型 | 执行时间/s |
进入大楼 | 0 | 30 |
等待电梯 | 1 | 依情况而定 |
等待进入房间 | 2 | 依情况而定 |
从房间走到房间 | 3 | 10 |
从电梯走到房间 | 4 | 10 |
从房间走到电梯 | 5 | 10 |
呆在房间中 | 6 | 输入数据 |
在电梯中 | 7 | 30/层 |
离开大楼 | 8 | 30 |
整个访问者访问流程中只有这9基本逻辑事件,访问者们根据自己的初始数据开始访问,访问过程中由于其他访问者的互相影响(即指同时访问电梯或房间时的排队现象)不断修改自己的访问计划,直至最终完成访问离开总部大楼。
//------------------------------------------------4th floor-------------------------------------------------- 【数据结构分析】
根据前述,不难看出本题重点在排队和访问者的访问清单。 由于排队是按照访问者的优先级别决定哪个靠前,因此可以使用以访问者优先级为判定标准的优先队列。 由于访问者的访问清单中欲访问房间的数目不明,因此使用链表保存各访问者的访问清单,一个节点保存该访问者一个事件(指上述9个事件中的一个)
//------------------------------------------------5th floor-------------------------------------------------- 【 题目关键细节分析】
细节①:如何处理电梯等待事件 由于本题电梯比较理想,因此我们只需要考虑访问者进入电梯的时间是否5的倍数,若不是,则需再等若干秒直到满足条件。 然而若有两名及以上访问者需要同时进入电梯,则需按照他们的优先级依次进入,因此排在后面的一个访问者就要多等上5秒钟(因为此时前一个优先级高的访问者刚进电梯,此时本访问者无法进入电梯,只能再等5秒)细节②:如何表示电梯 由于大楼中有房间与电梯两种物件,但实际上他们都是容器,用来装人的,因此可以把电梯视为一种特殊的房间,按照表示房间的方式表示即可,处理时按照处理电梯的方式处理即可。即房间号从01~10,我们用00号表示电梯,如0300表示3楼的电梯。
//------------------------------------------------6th floor--------------------------------------------------
【程序实现思路】
这里给出实现整个过程的思路,具体实现因人而异,我把这题当练习数据结构来做的,因此包括优先队列类在内的几乎所有函数模块都是自己编码实现,大大增加了代码量,很多都可以直接用STL或标准库函数。 ①事件表格的程序表示
//事件枚举类型 enum EventType
{
et_entry,
et_wait_elevator,
et_wait_front_room,
et_room2room,
et_elevator2room,
et_room2elevator,
et_in_room,
et_in_elevator,
et_exit
};
const size_t event_time[] = {30,5,0,10,10,10,0,30,30}; event_time[et_wait_front_room] 即为等待进入房间的初始时间,由于初始化时不会有访问者的排队情况,因此此项的值为0
②事件结构体及访问者事件链表 事件结构体拥有访问者标记,本事件开始时间,本事件持续时间,本事件的事件类型,本事件的起始地点,本事件的目的地点 六个数据项 (这里我做复杂了,string 类型可以直接用int型数据合理替换的,我为了表示方便,选择了string,有时间再改过) 访问者事件链表节点拥有事件数据域以及一个指向下一节点的指针 //事件结构体 struct Event
{
char _visitor_id; //访问者标号
string _start_time; //事件开始时间
size_t _duration_time; //事件持续时间
EventType _event_type; //事件类型
string _from; //起始地点
string _to; //目的地点
};
③由于整个流程只有9个逻辑事件,因此,我们独立出9个模块以便将对应事件追加到对应访问者的访问链表末尾,通过一个初始化过程,调用这9个模块,初始化每个访问者的访问链表。 //访问者行程表节点
struct Schedule
{
Schedule *next;
Event eve;
};
string Entry(string start_time, string id, Schedule *head);
string WaitElevator(string start_time, Schedule *head);
string WaitFrontRoom(string start_time, string to, Schedule *head);
string RoomToRoom(string start_time, string to, Schedule *head);
string ElevatorToRoom(string start_time, string to, Schedule *head);
string RoomToElevator(string start_time, string to, Schedule *head);
string InRoom(string start_time, string to, size_t duration_time, Schedule *head);
string InElevator(string start_time, string to, Schedule *head);
void Exit(string start_time, Schedule *head);
以上9个模块除了Exit可能(即访问者离开时不在1楼)调用WaitElevator以及RoomToElevator之外,其余模块都独立运行,将该模块对应的新事件节点追加到对应访问者的访问链表末尾,并返回本次事件的结束时间,以作为其余模块的参数传递(即下一个事件的开始时间)。
④初始化访问者链表 根据输入数据,初始化所有访问者的行程链表 //所有访问者存入一个vector,每个vector元素首为一个访问者访问链表头结点 void InitSchedule(vector<Schedule*> &visitors_schedule) ⑤处理初始化后的访问者链表 由于访问过程中可能存在等待事件,因此我们必须针对等待事件酌情更新等待之后的所有事件的开始时间,若遇到等待电梯事件,更新开始时间时还要考虑其是否5的倍数。 ⑥输出访问者的详细行程 这里强调下,由于题目并没有说输入数据是按字母顺序排列的,但是要求了输出的结果要按升序排列,因此输出时还要对vector中所有链表按其访问者标识字母由小到大排序后再输出 //------------------------------------------------7th floor--------------------------------------------------
【访问流程部分的源代码】
以下是整个流程的源代码,由于我用的string类型表示HH:MM:SS的时间,因此多了许多转换过程,以及其他辅助函数,这里就不一一列举了。
#include"event.h" //行程表节点比较函数 bool Less(void *x, void *y); //全体优先队列数组([有效楼层01~10,00号楼层为特殊值,不予计算][有效房间00~10]) PQueue<Schedule*,Less> queues[kFloorCount][kRoomCount]; /* * 模块功能:行程表节点的比较 * 模块返回:若x小于y,返回true,否则false */ bool Less(void *x, void *y) { if(TimeCmp((*(Schedule**)(x))->eve._start_time,(*(Schedule**)(y))->eve._start_time) == -1) return true; else if(TimeCmp((*(Schedule**)(x))->eve._start_time,(*(Schedule**)(y))->eve._start_time) == 0 && (*(Schedule**)(x))->eve._visitor_id < (*(Schedule**)(y))->eve._visitor_id) return true; return false; } /* * 模块功能:得到指向行程链表尾节点的指针 */ Schedule* GetLastNode(Schedule *head) { for( ;head->next != NULL; ) head = head->next; return head; } /* * 模块功能:根据访问者进入大楼事件设置访问者行程链表 * 模块返回:模块返回本次事件完成之后的时间,作为后续事件的开始时间 */ string Entry(string start_time, string id, Schedule *head) { Schedule *new_schedule = new Schedule; assert(new_schedule != NULL); new_schedule->next = NULL; new_schedule->eve._visitor_id = id[0]; new_schedule->eve._start_time = start_time; new_schedule->eve._duration_time = event_time[et_entry]; new_schedule->eve._event_type = et_entry; new_schedule->eve._from = "0000"; new_schedule->eve._to = "0000"; //将本事件追加到事件链表尾 GetLastNode(head)->next = new_schedule; //返回本次事件结束时间以作为后续事件的开始时间 return GetTime(new_schedule->eve._start_time,new_schedule->eve._duration_time); } /* * 模块功能:根据访问者等待电梯事件设置访问者行程链表 * 模块返回:模块返回本次事件完成之后的时间,作为后续事件的开始时间 */ string WaitElevator(string start_time, Schedule *head) { Schedule *new_schedule = new Schedule; assert(new_schedule != NULL); new_schedule->next = NULL; new_schedule->eve._visitor_id = head->next->eve._visitor_id; new_schedule->eve._start_time = start_time; new_schedule->eve._event_type = et_wait_elevator; new_schedule->eve._from = GetLastNode(head)->eve._to; if(new_schedule->eve._from == "0000") new_schedule->eve._from[1] = '1'; new_schedule->eve._to = new_schedule->eve._from; //计算等待时间 size_t waiting_time = event_time[et_wait_elevator]; size_t start_seconds = CStr2Int(start_time.substr(6,2).c_str()); if(start_seconds % waiting_time == 0) new_schedule->eve._duration_time = 0; else new_schedule->eve._duration_time = waiting_time - start_seconds % waiting_time; //将本事件追加到事件链表尾 GetLastNode(head)->next = new_schedule; //将本次电梯等待事件加入对应楼层的电梯等待队列中 queues[CStr2Int(new_schedule->eve._from.substr(0,2).c_str())][0].push(new_schedule); //返回本次事件结束时间以作为后续事件的开始时间 return GetTime(new_schedule->eve._start_time,new_schedule->eve._duration_time); } /* * 模块功能:根据访问者等待进入房间事件设置访问者行程链表 * 模块返回:模块返回本次事件完成之后的时间,作为后续事件的开始时间 */ string WaitFrontRoom(string start_time, string to, Schedule *head) { Schedule *new_schedule = new Schedule; assert(new_schedule != NULL); new_schedule->next = NULL; new_schedule->eve._visitor_id = head->next->eve._visitor_id; new_schedule->eve._start_time = start_time; new_schedule->eve._duration_time = event_time[et_wait_front_room]; new_schedule->eve._event_type = et_wait_front_room; new_schedule->eve._from = GetLastNode(head)->eve._to; new_schedule->eve._to = to; //将本事件追加到事件链表尾 GetLastNode(head)->next = new_schedule; //将本事件加入对应房间等待队列 queues[CStr2Int(to.substr(0,2).c_str())][CStr2Int(to.substr(2,2).c_str())].push(new_schedule); return GetTime(new_schedule->eve._start_time,new_schedule->eve._duration_time); } /* * 模块功能:根据访问者从同一楼层的一个房间走到另一个房间事件设置访问者行程链表 * 模块返回:模块返回本次事件完成之后的时间,作为后续事件的开始时间 */ string RoomToRoom(string start_time, string to, Schedule *head) { Schedule *new_schedule = new Schedule; assert(new_schedule != NULL); new_schedule->next = NULL; new_schedule->eve._visitor_id = head->next->eve._visitor_id; new_schedule->eve._start_time = start_time; new_schedule->eve._event_type = et_room2room; new_schedule->eve._from = GetLastNode(head)->eve._to; new_schedule->eve._to = to; new_schedule->eve._duration_time = event_time[et_room2room]; //将本事件追加到事件链表尾 GetLastNode(head)->next = new_schedule; return GetTime(new_schedule->eve._start_time,new_schedule->eve._duration_time); } /* * 模块功能:根据访问者从同一楼层的电梯里走到房间事件设置访问者行程链表 * 模块返回:模块返回本次事件完成之后的时间,作为后续事件的开始时间 */ string ElevatorToRoom(string start_time, string to, Schedule *head) { Schedule *new_schedule = new Schedule; assert(new_schedule != NULL); new_schedule->next = NULL; new_schedule->eve._visitor_id = head->next->eve._visitor_id; new_schedule->eve._start_time = start_time; new_schedule->eve._event_type = et_elevator2room; new_schedule->eve._from = GetLastNode(head)->eve._to; new_schedule->eve._from[2] = '0'; new_schedule->eve._from[3] = '0'; new_schedule->eve._to = to; new_schedule->eve._duration_time = event_time[et_elevator2room]; //将本事件追加到事件链表尾 GetLastNode(head)->next = new_schedule; return GetTime(new_schedule->eve._start_time,new_schedule->eve._duration_time); } /* * 模块功能:根据访问者从同一楼层的房间里走到电梯中事件设置访问者行程链表 * 模块返回:模块返回本次事件完成之后的时间,作为后续事件的开始时间 */ string RoomToElevator(string start_time, string to, Schedule *head) { Schedule *new_schedule = new Schedule; assert(new_schedule != NULL); new_schedule->next = NULL; new_schedule->eve._visitor_id = head->next->eve._visitor_id; new_schedule->eve._start_time = start_time; new_schedule->eve._event_type = et_room2elevator; new_schedule->eve._from = GetLastNode(head)->eve._to; new_schedule->eve._to = to; new_schedule->eve._duration_time = event_time[et_room2elevator]; //将本事件追加到事件链表尾 GetLastNode(head)->next = new_schedule; return GetTime(new_schedule->eve._start_time,new_schedule->eve._duration_time); } /* * 模块功能:根据访问者从同一楼层的房间里走到电梯中事件设置访问者行程链表 * 模块返回:模块返回本次事件完成之后的时间,作为后续事件的开始时间 */ string InRoom(string start_time, string to, size_t duration_time, Schedule *head) { Schedule *new_schedule = new Schedule; assert(new_schedule != NULL); new_schedule->next = NULL; new_schedule->eve._visitor_id = head->next->eve._visitor_id; new_schedule->eve._start_time = start_time; new_schedule->eve._event_type = et_in_room; new_schedule->eve._from = GetLastNode(head)->eve._to; new_schedule->eve._to = to; new_schedule->eve._duration_time = duration_time; //将本事件追加到事件链表尾 GetLastNode(head)->next = new_schedule; return GetTime(new_schedule->eve._start_time,duration_time); } /* * 模块功能:根据访问者从同一楼层的房间里走到电梯中事件设置访问者行程链表 * 模块返回:模块返回本次事件完成之后的时间,作为后续事件的开始时间 */ string InElevator(string start_time, string to, Schedule *head) { Schedule *new_schedule = new Schedule; assert(new_schedule != NULL); new_schedule->next = NULL; new_schedule->eve._visitor_id = head->next->eve._visitor_id; new_schedule->eve._start_time = start_time; new_schedule->eve._event_type = et_in_elevator; string from = new_schedule->eve._from = GetLastNode(head)->eve._to; new_schedule->eve._to = to; if(from == "0000") from[1] = '1'; //计算在电梯里呆的时间 new_schedule->eve._duration_time = (abs(CStr2Int(new_schedule->eve._to.substr(0,2).c_str()) - CStr2Int(from.substr(0,2).c_str())) ) * event_time[et_in_elevator]; //将本事件追加到事件链表尾 GetLastNode(head)->next = new_schedule; return GetTime(new_schedule->eve._start_time,new_schedule->eve._duration_time); } /* * 模块功能:根据访问者离开大楼事件设置访问者行程链表 */ void Exit(string start_time, Schedule *head) { Schedule *new_schedule = new Schedule; assert(new_schedule != NULL); new_schedule->next = NULL; new_schedule->eve._visitor_id = head->next->eve._visitor_id; new_schedule->eve._duration_time = event_time[et_exit]; new_schedule->eve._event_type = et_exit; new_schedule->eve._from = GetLastNode(head)->eve._to; //若访问者此时不在1楼,计算其到达1楼的时间 int floor = CStr2Int(new_schedule->eve._from.substr(0,2).c_str()); if(floor != 1 && floor != 0) { string elevator = new_schedule->eve._from.substr(0,2); elevator += "00"; start_time = RoomToElevator(start_time,elevator,head); start_time = WaitElevator(start_time,head); start_time = InElevator(start_time,"0100",head); } new_schedule->eve._start_time = start_time; //将本事件追加到事件链表尾 GetLastNode(head)->next = new_schedule; } /* * 模块功能:根据input文件初始化所有访问者访问事件链表 */ void InitSchedule(vector<Schedule*> &visitors_schedule) { ifstream fin("input.txt",ofstream::in); if(!fin) exit(1); while(true) {//1次循环初始化完成1个访问者行程链表 string id; string start_time; fin>>id; if(id == ".") break; fin>>start_time; Schedule *head = new Schedule; assert(head != NULL); head->next = NULL; visitors_schedule.push_back(head); head = visitors_schedule.back(); start_time = Entry(start_time,id,head); while(true) {//构建本访问者的访问事件链表 string room; size_t duration_time; fin>>room; if(room == "0") { Exit(start_time,head); break; } fin>>duration_time; Schedule *cur_schedule = GetLastNode(head); if(cur_schedule->eve._to.substr(0,2) == room.substr(0,2) || ((cur_schedule->eve._event_type == et_entry) && room.substr(0,2) == "01")) {//当前位置与目的地同层 if(!(cur_schedule->eve._event_type == et_entry)) {//并非刚进入大楼 start_time = RoomToRoom(start_time,room,head); } } else {//当前位置与目的地不同层 if(!(cur_schedule->eve._event_type == et_entry)) {//并非刚进入大楼 cur_schedule = GetLastNode(head); string elevator = cur_schedule->eve._to.substr(0,2); elevator += "00"; start_time = RoomToElevator(start_time,elevator,head); } start_time = WaitElevator(start_time,head); start_time = InElevator(start_time,room,head); start_time = ElevatorToRoom(start_time,room,head); } start_time = WaitFrontRoom(start_time,room,head); start_time = InRoom(start_time,room,duration_time,head); } } } /* * 模块功能:将行程表向量以访问者优先级由高到低重新排序 */ void SortById(vector<Schedule*> &visitors_schedule) { assert(visitors_schedule.size() > 0); for(vector<Schedule*>::size_type i=0; i!=visitors_schedule.size()-1; ++i) { vector<Schedule*>::size_type j = i + 1; Schedule* key = visitors_schedule[j]; while(j>0 && visitors_schedule[j-1]->next->eve._visitor_id > key->next->eve._visitor_id) { visitors_schedule[j] = visitors_schedule[j-1]; --j; } visitors_schedule[j] = key; } } /* * 模块功能:销毁行程表 */ void DestroySchedule(vector<Schedule*> &visitors_schedule) { for(vector<Schedule*>::iterator it=visitors_schedule.begin(); it != visitors_schedule.end(); ++it) delete (*it); visitors_schedule.clear(); } /* * 模块功能:将所有访问者的行程表输出到文件 */ void OutputSchedule(vector<Schedule*> &visitors_schedule) { //各种事件消息 const string msg[] = {"Entry", "Waiting in elevator queue", "Waiting in front of room xxxx", "Transfer from room xxxx to room xxxx", "Transfer from elevator to room xxxx", "Transfer from room xxxx to elevator", "Stay in room xxxx", "Stay in elevator", "Exit" }; ofstream fout("output",ofstream::out); //将行程表向量以访问者优先级由高到低重新排序 SortById(visitors_schedule); vector<Schedule*>::size_type index = 0; while(true) { if(index == visitors_schedule.size()) break; Schedule *idx = visitors_schedule[index]->next; fout<<idx->eve._visitor_id<<endl; while(true) { if(idx == NULL) break; if(idx->eve._start_time == GetTime(idx->eve._start_time,idx->eve._duration_time)) { idx = idx->next; continue; } fout<<idx->eve._start_time<<' ' <<GetTime(idx->eve._start_time,idx->eve._duration_time)<<' '; EventType type = idx->eve._event_type; string message = msg[type]; string room; if(type == et_wait_front_room) { room = idx->eve._to; message.erase(25); message += room; } else if(type == et_room2room) { room = idx->eve._from; message.erase(19,4); message.insert(19,room); room = idx->eve._to; message.erase(32); message += room; } else if(type == et_elevator2room) { room = idx->eve._to; message.erase(31); message += room; } else if(type == et_room2elevator) { //Transfer from room xxxx to elevator room = idx->eve._from; message.erase(19,4); message.insert(19,room); } else if(type == et_in_room) { room = idx->eve._to; message.erase(13); message += room; } fout<<message<<endl; idx = idx->next; } fout<<endl; ++index; } } /* * 模块功能:行程表处理过程,根据各访问者的原始行程表,考虑他们之间的相互影响 * 形成所有访问者的最终行程表 */ void ProcessSchedule(void) { //队首优先队列,用于得到所有队首中最小者 PQueue<Schedule*,Less> queues_top; //保存上一次出队的队首的矩阵 Schedule *pre_top[kFloorCount][kRoomCount] = {NULL}; //得到所有队列中共有多少个事件 size_t total_event = 0; for(size_t i=1; i!=kFloorCount; ++i) for(size_t j=0; j!=kRoomCount; ++j) total_event += queues[i][j].size(); //调整所有访问者的行程表,直到所有等待队列为空 while(true) { if(total_event == 0) break; //将所有队首入优先队列,从而得到当前所有队首的最小者 queues_top.clear(); for(size_t i=1; i!=kFloorCount; ++i) for(size_t j=0; j!=kRoomCount; ++j) if(!queues[i][j].empty()) queues_top.push(queues[i][j].top()); //计算最小事件在队列数组中的下标 size_t floor = CStr2Int(queues_top.top()->eve._to.substr(0,2).c_str()); size_t room = CStr2Int(queues_top.top()->eve._to.substr(2,2).c_str()); //事件时间调整标记,未经调整为false bool delayed = false; //若本队列不是第一次出队,更新本次等待事件的延迟时间 Schedule *pre_event = pre_top[floor][room]; if(pre_event != NULL) { Schedule *cur_event = queues_top.top(); //本次最小事件是等待电梯 if(cur_event->eve._event_type == et_wait_elevator) { //若本次等待电梯事件的开始时间与上一次队首的开始时间相同,则需再等5秒 if(cur_event->eve._start_time == pre_event->eve._start_time) { cur_event->eve._duration_time = pre_event->eve._duration_time + event_time[et_wait_elevator]; delayed = true; } } else //最小事件是等待进入房间,更新此事件的等待时间 { //计算上一个队首离开房间的时间 string exit_room_time = GetTime(pre_event->next->eve._start_time, pre_event->next->eve._duration_time); int time_sub = TimeSub(exit_room_time,GetTime(cur_event->eve._start_time, cur_event->eve._duration_time)); //上一次的队首仍在房中 if(time_sub > 0) { cur_event->eve._duration_time += time_sub; delayed = true; } } //只要延时了,则维护本次事件之后的所有事件的时间连贯性,并维护对应堆性质 if(delayed) { //算法描述:本次事件之后的事件的开始时间依次加上上一次事件的等待时间 // 若下一次事件是电梯等待事件,则还要调整本次事件的等待时间 // 以满足本次事件结束(即进入电梯)的时间是5的倍数 while(true) { //得到下一事件 Schedule *next_event = cur_event->next; if(next_event == NULL) break; //更新下一事件的开始时间 next_event->eve._start_time = GetTime(cur_event->eve._start_time,cur_event->eve._duration_time); //若下一次次事件是电梯等待事件,则还需需更新电梯等待时间,以满足5秒要求 if(next_event->eve._event_type == et_wait_elevator) { //重新计算下一事件的等待时间 next_event->eve._duration_time = 0; size_t waiting_time = event_time[et_wait_elevator]; size_t start_seconds = CStr2Int(next_event->eve._start_time.substr(6,2).c_str()); if(start_seconds % waiting_time != 0) next_event->eve._duration_time = (waiting_time - (start_seconds % waiting_time)); } //计算本事件所在楼层即房间 size_t floor = CStr2Int(cur_event->eve._to.substr(0,2).c_str()); size_t room = CStr2Int(cur_event->eve._to.substr(2,2).c_str()); //维护本事件所在队列的堆性质 queues[floor][room].heap_ify(); //更新当前事件 cur_event = cur_event->next; } } } //该最小事件出队,并以此更新pre_top对应元素 pre_top[floor][room] = queues[floor][room].pop(); --total_event; } } /* * 模块功能:模拟整个访问过程 */ void Visiting(void) { //新建所有访问者的行程表 vector<Schedule*> visitors_schedule; //初始化所有行程链表 InitSchedule(visitors_schedule); //依据各访问者之间的相互影响,处理初始化后的访问者行程表 ProcessSchedule(); //将行程表输出到文件 OutputSchedule(visitors_schedule); //销毁行程链表 DestroySchedule(visitors_schedule); }