BUAA-OO-电梯调度
BUAA-OO-电梯调度
1.设计策略
需求分析
设计一个系统,使其可以根据乘客的当前楼层和目的楼层,为乘客分配电梯资源并运送其至目的楼层。
自顶向下
根据需求,可以将整个系统分成三个部分:
- 处理乘客的当前楼层和目的楼层,即输入处理
- 为乘客分配电梯资源,即调度
- 运送至目的楼层,即电梯本体
显然,这三个部分是有次序关系的:乘客输入,处理输入,将请求放入调度队列,调度器从调度队列中取出请求,根据各电梯状态将请求分配给合适的电梯。
上述的过程可以看成是三组 Producer-Consumer 组:
- Producer: 乘客输入,Consumer: 输入处理
- Producer: 输入处理,Consumer: 调度器
- Producer: 调度器,Consumer: 电梯
输入处理
阻塞式获取输入,直到检测到 EOF 或者输入流关闭。处理后的请求存储在 PersonRequest 类中。当获取到合法请求,将请求存入调度队列,唤醒调度器。
调度器
从调度队列中取出请求,根据电梯状态来决定请求被分配至的电梯,对于需要换乘的请求,将后续请求也分配至同一电梯,待前置请求完成,重新对后续请求进行调度。这里的电梯状态包括但不止于电梯能够停靠的楼层、电梯当前运行状态、电梯当前的任务序列、电梯历史运行状态。前两者对于调度起着决定性的作用,而后两者主要是使得各电梯负载均衡。
电梯
从任务序列中取出任务,前往指定楼层执行任务。若完成了某个需换乘请求的前置请求,将后续请求提交给调度器,使其被重新调度。
2.具体实现
类的设置
elevator
├── Direction
├── Dispatcher
├── DivideTask
├── Elevator
├── ElevatorRequest
├── InputHandler
├── Main
├── RequestQueue
├── Task
└── Timer
InputHandler
顾名思义,用于处理乘客输入的类。域中包含有一个调度器(Dispatcher 类实例)。
当获取到合法输入时,唤醒(notify)调度器,否则堵塞。
当输入为 EOF 或者流被关闭时,将调度器的状态设置为停止状态(false),并唤醒(notify)调度器。
Dispatcher
调度器类。域中包含有
- 一个调度队列(RequestQueue 类实例)。
- 各个电梯(Elevator 类实例)。
- 各个电梯的部分状态:能够停靠的楼层、电梯历史运行状态。
当调度队列为空,进入等待状态(wait)。
被唤醒后,从调度队列中依次取出所有请求并进行分配。全部取出后,若状态为运行状态(true),重新进入等待状态(wait);否则,将所有电梯的状态设置为停滞状态(false),并依次唤醒,最后结束调度器线程,但是保留调度器实例。
Elevator
电梯类。域中包含有
- 电梯状态,包括开关门、运行速度,当前楼层,当前目标楼层。
- 两个 Direction 类实例,一个用于表示当前的任务状态(向上或是向下),一个用于表示当前的运行方向。
- 两个 ElevatorRequest 类实例,一个用于存储向上(目标楼层大于出发楼层)的任务,一个用于存储向下(目标楼层小于出发楼层)的任务。
- 一个 LinkedList<DivideTask>类实例,用于存储将要被重新调度的后续请求。
电梯运行总体采用 Look 算法:在不转向的前提下先执行完毕一个方向的任务,然后切换为另一个方向的任务序列,继续执行。执行过程中的等待过程都通过 TImer 类来实现。
Task
任务类。单纯的数据聚合。域中包含有
- 存储任务方向的字符串("IN","OUT")
- 乘客 ID
- 目标楼层
ElevatorRequest
电梯任务类。域中包含有
- 两个 LinkedList<Task>类实例,一个是主任务序列,另一个是副任务序列。
- 一个 Direction 类实例,用于表示任务序列的方向。
当电梯接收到请求,会将请求存入 ElevatorRequest 中。首先进行方向的判断,并将其拆分为两个任务:一个 IN,一个 OUT。然后判断加入主任务序列或是加入副任务序列:若与电梯的任务方向相反,或者可以捎带,则加入主任务序列,否则加入副任务序列。
当主任务序列为空,电梯转向时,进行 fresh 操作,将主任务序列与副任务序列互换。
DivideRequest
待调度请求类。单纯的数据聚合。域中包含有
- 请求(PersonRequest 类实例)
- 目标电梯
3.类图与复杂度分析
类图
如下,与上述的设计描述大致相同。

复杂度
使用 Complexity mertics 生成。

总体而言,复杂度尚可,复杂度较高的几个 method 都是与设计紧密相关的,没有修改或者拆分的必要。
4.自身 bug
在第二次电梯作业中,由于电梯的上下极限从 1-15 变为-3-20,而我在第一次作业中空任务的标志 Magic Number 设置为了 0 忘记修改,导致第二次作业出现了问题。由此可以看出测试的重要性,因而第三次作业我专门写了一个评测机放在服务器上进行长时间测试。
5.心得体会
在多线程编程中,要特别关注暴露在多个线程之外的副作用,严格控制进入临界区的线程数量。
BUAA-OO-电梯调度的更多相关文章
- OO电梯调度
告别了三次奇妙无比的求导作业之后,我们就开始搭建一部自己的电梯了.相信我们不同同学的电梯运行方式肯定各具特色吧,但值得肯定的是,在艰苦的走完了三次电梯逐步改进的作业之后,我们的电梯在正常情况下应该是可 ...
- OO第二次博客作业——电梯调度
OO第二次博客作业——电梯调度 前言 最近三周,OO课程进入多线程学习阶段,主要通过三次电梯调度作业来学习.从单部电梯的傻瓜式调度到有性能要求的调度到多部电梯的调度,难度逐渐提升,对同学们的要求逐渐变 ...
- OO第二单元总结(多线程的电梯调度)
经过第一单元作业的训练,在做第二单元的作业的时候,要更加的有条理.但是第二次作业多线程的运行,带来了更多的运行的不确定性.呈现出来就是程序会出现由于线程安全问题带来的不可复现的bug.本单元的作业也让 ...
- oo第二次博客-三次电梯调度的总结与反思
本单元从电梯调度相关问题层层深入,带领我们学习并运用了了多线程相关的知识. 三次电梯调度依次为单电梯单容量.单电梯可携带.多电梯可携带. 一.我的设计 在第一次作业中,使用了最简单的FIFO调度方法. ...
- 「BUAA OO Unit 2 HW8」第二单元总结
「BUAA OO Unit 2 HW8」第二单元总结 目录 「BUAA OO Unit 2 HW8」第二单元总结 Part 0 前言 Part 1 第五次作业 1.1 作业要求 1.2 架构设计 1. ...
- 「BUAA OO Unit 4 HW16」第四单元总结与课程回顾
「BUAA OO Unit 4 HW16」第四单元总结与课程回顾 目录 「BUAA OO Unit 4 HW16」第四单元总结与课程回顾 Part 0 第四单元作业架构设计 架构设计概要 AppRun ...
- 电梯调度编写(oo-java编程)
第二单元的问题是写一个关于电梯调度的程序. 需要模拟一个多线程实时电梯系统,从标准输入中输入请求信息,程序进行接收和处理,模拟电梯运行,将必要的运行信息通过输出接口进行输出. 主要锻炼学生的多线程程序 ...
- BUAA OO 2019 第二单元作业总结
目录 总 架构 controller model view 优化算法 Look 算法 多种算法取优 预测未来 多线程 第五次作业 第六次作业 第七次作业 代码静态分析 UML 类图 类复杂度 类总代码 ...
- 学会拒绝,是一种智慧——OO电梯章节优化框架的思考
在本章的三次作业里,每次作业我都有一个主题,分别是:托盘型共享数据.单步电梯运行优化.多部电梯运行优化,因而电梯优化实际是第二.三次作业.虽然后两次作业从性能分上看做得还不错,但阅读其他大佬博客,我深 ...
- 【软件工程】电梯调度的初步实现 李亚文&&郭莉莉
一.开门见山,代码粘 using System; using System.Collections.Generic; using System.Data; using System.Drawing; ...
随机推荐
- 总结JAVA----IO流中的字节流
对于IO流中字节流的总结 字节流的概念 由于应用程序,经常需要和文件打交道,所以Inputstream专门提供了读写文件的子类:FileInputStream和FileOutputStream类,如果 ...
- Hybrid APP之Native和H5页面交互原理
Hybrid APP之Native和H5页面交互原理 Hybrid APP的关键是原生页面与H5页面直接的交互,如下图,痛过JSBridge,H5页面可以调用Native的api,Native也可调用 ...
- Rsync + sersync 实时同步备份
一 Rsync + Sersync 实时同步介绍 1.Rsync 服务搭建介绍 云机上搭建Rsync server,在本地搭建Rsync Clinet. 2. Sersync 服务搭建介绍 ...
- 【原创】互联网项目中mysql应该选什么事务隔离级别
摘要 企业千万家,靠谱没几家. 社招选错家,亲人两行泪. 祝大家金三银四跳槽顺利! 引言 开始我们的内容,相信大家一定遇到过下面的一个面试场景 面试官:"讲讲mysql有几个事务隔离级别?& ...
- [题解]NOIP2018(普及组)T1标题统计(title)
NOIP2018(普及组)T1标题统计(title) 题解 [代码(AC)] #include <iostream> #include <cstdio> #include &l ...
- Vue—组件传值及vuex的使用
一.父子组件之间的传值 1.父组件向子组件传值: 子组件在props中创建一个属性,用以接收父组件传来的值 父组件中注册子组件 在子组件标签中添加子组件props中创建的属性 把需要传给子组件的值赋给 ...
- axios delete请求 参数
内容转载作者:https://www.cnblogs.com/pangguoming/p/8994742.html 如果服务端将参数当做 java对象来封装接收则 参数格式为:{data: param ...
- python-装饰器的最终形态和固定格式 语法糖
import time def timer(f): # 这是一个装饰器函数 def inner(): start = time.time() f() # 被装饰的函数 end = time.time( ...
- Laravel数据库迁移
Laravel的数据迁移功能很好用,并且可以带来一系列好处.通过几条简单的 artisan 命令,就可以顺利上手,没有复杂的地方 注意:该系列命令对数据库非常危险,请准备一个单独的数据库作为配套练习, ...
- Manual write code to record error log in .net by Global.asax
完整的Glabal.asax代码: <%@ Application Language="C#" %> <script RunAt="server&quo ...