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-电梯调度的更多相关文章

  1. OO电梯调度

    告别了三次奇妙无比的求导作业之后,我们就开始搭建一部自己的电梯了.相信我们不同同学的电梯运行方式肯定各具特色吧,但值得肯定的是,在艰苦的走完了三次电梯逐步改进的作业之后,我们的电梯在正常情况下应该是可 ...

  2. OO第二次博客作业——电梯调度

    OO第二次博客作业——电梯调度 前言 最近三周,OO课程进入多线程学习阶段,主要通过三次电梯调度作业来学习.从单部电梯的傻瓜式调度到有性能要求的调度到多部电梯的调度,难度逐渐提升,对同学们的要求逐渐变 ...

  3. OO第二单元总结(多线程的电梯调度)

    经过第一单元作业的训练,在做第二单元的作业的时候,要更加的有条理.但是第二次作业多线程的运行,带来了更多的运行的不确定性.呈现出来就是程序会出现由于线程安全问题带来的不可复现的bug.本单元的作业也让 ...

  4. oo第二次博客-三次电梯调度的总结与反思

    本单元从电梯调度相关问题层层深入,带领我们学习并运用了了多线程相关的知识. 三次电梯调度依次为单电梯单容量.单电梯可携带.多电梯可携带. 一.我的设计 在第一次作业中,使用了最简单的FIFO调度方法. ...

  5. 「BUAA OO Unit 2 HW8」第二单元总结

    「BUAA OO Unit 2 HW8」第二单元总结 目录 「BUAA OO Unit 2 HW8」第二单元总结 Part 0 前言 Part 1 第五次作业 1.1 作业要求 1.2 架构设计 1. ...

  6. 「BUAA OO Unit 4 HW16」第四单元总结与课程回顾

    「BUAA OO Unit 4 HW16」第四单元总结与课程回顾 目录 「BUAA OO Unit 4 HW16」第四单元总结与课程回顾 Part 0 第四单元作业架构设计 架构设计概要 AppRun ...

  7. 电梯调度编写(oo-java编程)

    第二单元的问题是写一个关于电梯调度的程序. 需要模拟一个多线程实时电梯系统,从标准输入中输入请求信息,程序进行接收和处理,模拟电梯运行,将必要的运行信息通过输出接口进行输出. 主要锻炼学生的多线程程序 ...

  8. BUAA OO 2019 第二单元作业总结

    目录 总 架构 controller model view 优化算法 Look 算法 多种算法取优 预测未来 多线程 第五次作业 第六次作业 第七次作业 代码静态分析 UML 类图 类复杂度 类总代码 ...

  9. 学会拒绝,是一种智慧——OO电梯章节优化框架的思考

    在本章的三次作业里,每次作业我都有一个主题,分别是:托盘型共享数据.单步电梯运行优化.多部电梯运行优化,因而电梯优化实际是第二.三次作业.虽然后两次作业从性能分上看做得还不错,但阅读其他大佬博客,我深 ...

  10. 【软件工程】电梯调度的初步实现 李亚文&&郭莉莉

    一.开门见山,代码粘 using System; using System.Collections.Generic; using System.Data; using System.Drawing; ...

随机推荐

  1. windows10 uwp获取设备当前地理位置(经纬度)

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...

  2. Cookie Session和自定义分页

    cookie Cookie的由来 大家都知道HTTP协议是无状态的. 无状态的意思是每次请求都是独立的,它的执行情况和结果与前面的请求和之后的请求都无直接关系,它不会受前面的请求响应情况直接影响,也不 ...

  3. HashMap、Hashtable、ConcurrentHashMap的原理与区别

    同步首发:http://www.yuanrengu.com/index.php/2017-01-17.html 如果你去面试,面试官不问你这个问题,你来找我^_^ 下面直接来干货,先说这三个Map的区 ...

  4. django开发新手教程(原创)

    为了帮助新手简单高效解决django开发的问题,从而写了这么一篇,随便转载! 本人用的是windows10操作系统 #联网安装 ==指定版本号      我在自定义的www文件夹安装C:\Users\ ...

  5. JSP中常用的的EL表达式的汇总

    Jsp基础知识 jsp的组成 html静态页面(css.javascript) java代码 <% %> (_jspService方法中) 内置对象 out request 表达式 < ...

  6. ios之库Protobuf的使用

    https://blog.csdn.net/dangbai01_/article/details/81099001 (1)Protobuf是什么? Protobuf 即 google protocol ...

  7. 在自己写的C#类中调用 ASP.NET的Request,server 等对象

    加命名空间(可能需要在项目中先加引用,再在类中如下引用) using System.Web.SessionState; HttpContext.Current.Session["TotalP ...

  8. 【MySQL 读书笔记】当我们在使用索引的时候我们在做什么

    我记得之前博客我也写过关于索引使用的文章,但是并不全面,这次尽量针对重点铺全面一点. 因为索引是数据引擎层的结构我们只针对最常见使用的 Innodb 使用的 B+Tree 搜索树结构进行介绍. 每一个 ...

  9. servlet(2)servlet过滤器

    1.servlet过滤器 用于动态的拦截servlet请求或响应,以变更或使用其中的信息. (1)过滤器和servlet是多对多的关系,即一个过滤器可以用于一个或多个servlet,多个过滤器也可以用 ...

  10. 启动django时报错Watching for file changes with StatReloader(使用状态加载程序监视文件更改 )

    原因:可能是Django版本和Python版本或者PyMysql版本不一致 解决:升级或者降级Django版本 命令如下: pip install django==2.1.7 #django==版本号 ...