当前位置: 首页 > news >正文

【设计】855. 考场就座

855. 考场就座

这段代码实现了一个考场安排座位的算法。在这个算法中,考场被模拟成一个从0到n-1的数轴,其中每个位置代表一个座位。目的是在每次学生入座时,找到一个使得所有学生之间距离最大化的座位,并在学生离开时更新座位信息。下面是具体的实现思路和算法描述:

  1. 初始化 (public ExamRoom(int n)):

    • n:考场座位的总数。
    • intervals:一个TreeSet,用于存储当前所有空闲区间,并按照某种规则排序。这里的排序规则首先比较两个区间可坐的最远距离,如果相同,则比较区间的末端位置。
    • startToIntervalendToInterval:两个HashMap,分别用于快速定位一个区间的开始和结束位置对应的区间对象。
    • 在初始化时,会添加一个特殊区间[-1, n]intervals中,表示整个考场最开始时全部为空。
  2. 入座 (public int seat()):

    • intervals中取出最优区间(即开始时的排序规则确定的第一个区间),然后根据该区间的起始位置和结束位置计算出新的座位位置。
    • 如果该区间的起始位置是-1,说明最左侧是空的,因此新的座位位置是0。
    • 如果该区间的结束位置是n,说明最右侧是空的,因此新的座位位置是n-1
    • 否则,新的座位位置是区间中点。
    • 入座后,将原区间分为两个新区间,并加入到intervals中。
  3. 离开 (public void leave(int p)):

    • 当一个学生离开座位时,通过endToIntervalstartToInterval找到该座位所属的两个区间。
    • 将这两个区间合并为一个新区间,并更新到intervals中。
  4. 辅助方法:

    • private void removeInterval(int[] interval):从intervalsstartToIntervalendToInterval中移除指定区间。
    • private void addInterval(int[] interval):向intervalsstartToIntervalendToInterval中添加新区间。
    • private int dist(int[] interval):计算一个区间中可以坐的最远距离。如果区间的起始或结束是边界,距离计算方式略有不同。

通过上述方法,该算法能够有效地在每次学生入座时找到最优的座位,并在学生离开时更新座位信息,以保证考场的座位安排尽可能地公平和高效。

class ExamRoom {int n;TreeSet<int[]> intervals;HashMap<Integer, int[]> startToInterval;HashMap<Integer, int[]> endToInterval;public ExamRoom(int n) {this.n = n;startToInterval = new HashMap<>();endToInterval = new HashMap<>();intervals = new TreeSet<>((a, b) ->dist(b)  != dist(a) ?  dist(b) - dist(a) : a[1] - b[1]);addInterval(new int[]{-1, n});}public int seat() {int[] interval  = intervals.pollFirst();int start = interval[0];int end = interval[1];int seat = 0;if (start == -1) {seat = 0;} else if (end == n) {seat = n - 1;} else {seat = (start + end) >>> 1;}addInterval(new int[]{start, seat});addInterval(new int[]{seat, end});return seat;}public void leave(int p) {int[] i1 = endToInterval.get(p);int[] i2 = startToInterval.get(p);int start = i1[0];int end = i2[1];int[] interval = new int[]{start, end};removeInterval(i1);removeInterval(i2);addInterval(interval);}private void removeInterval(int[] interval) {intervals.remove(interval);startToInterval.remove(interval[0]);endToInterval.remove(interval[1]);}private void addInterval(int[] interval) {intervals.add(interval);startToInterval.put(interval[0], interval);endToInterval.put(interval[1], interval);}private int dist(int[] interval) {int start = interval[0];int end = interval[1];if (start == -1 || end == n) {return end - start - 1;}return (end - start) / 2;}
}/*** Your ExamRoom object will be instantiated and called as such:* ExamRoom obj = new ExamRoom(n);* int param_1 = obj.seat();* obj.leave(p);*/
http://www.lryc.cn/news/313603.html

相关文章:

  • Android中的传感器类型和接口名称
  • 解析进程 /proc/pid/maps 和 /proc/pid/smaps
  • 【MQ】消息队列概述
  • 交友盲盒系统PHP开源的盲盒源码
  • 【Flutter 面试题】什么是异步编程 Flutter中如何处理异步操作?
  • 处理error: remote origin already exists.及其Gitee文件上传保姆级教程
  • 网络编程套接字(2)——Socket套接字
  • 向量错题本
  • FPGA-VGA成像原理与时序
  • 【VTKExamples::Points】第三期 ExtractClusters
  • 迅速上手:CentOS 系统下 SSH 服务配置指南
  • day38 动态规划part1
  • 01背包问题 刷题笔记
  • docker安装包(Linux和windows)
  • RabbitMQ 安装使用
  • echarts x轴名称过长tip显示全称
  • js和css阻塞问题
  • MySQL 的基础操作
  • 【python进阶篇】面向对象编程(1)
  • 力扣面试经典150 —— 6-10题
  • [密码学]入门篇——加密方式
  • 构建前后端分离项目常用的代码
  • 2575. 找出字符串的可整除数组(Go语言)
  • Redis与 Memcache区别
  • #QT(智能家居界面-界面切换)
  • js拓展-内置对象
  • 【李沐精读系列】GPT、GPT-2和GPT-3论文精读
  • Libevent的使用及reactor模型
  • 查看Linux服务器配置
  • 【机器学习】包裹式特征选择之递归特征添加法