携手创作,共同成长!这是我参与「掘金日新计划 · 8 月更文挑战」的第17天,点击查看活动详情 >>
座位预约管理系统
请你设计一个管理 n 个座位预约的系统,座位编号从 1 到 n 。
请你实现 SeatManager 类:
- SeatManager(int n) 初始化一个 SeatManager 对象,它管理从 1 到 n 编号的 n 个座位。所有座位初始都是可预约的。
- int reserve() 返回可以预约座位的 最小编号 ,此座位变为不可预约。
- void unreserve(int seatNumber) 将给定编号 seatNumber 对应的座位变成可以预约。
示例1:
输入:
["SeatManager", "reserve", "reserve", "unreserve", "reserve", "reserve", "reserve", "reserve", "unreserve"]
[[5], [], [], [2], [], [], [], [], [5]]
输出:
[null, 1, 2, null, 2, 3, 4, 5, null]
解释:
SeatManager seatManager = new SeatManager(5); // 初始化 SeatManager ,有 5 个座位。
seatManager.reserve(); // 所有座位都可以预约,所以返回最小编号的座位,也就是 1 。
seatManager.reserve(); // 可以预约的座位为 [2,3,4,5] ,返回最小编号的座位,也就是 2 。
seatManager.unreserve(2); // 将座位 2 变为可以预约,现在可预约的座位为 [2,3,4,5] 。
seatManager.reserve(); // 可以预约的座位为 [2,3,4,5] ,返回最小编号的座位,也就是 2 。
seatManager.reserve(); // 可以预约的座位为 [3,4,5] ,返回最小编号的座位,也就是 3 。
seatManager.reserve(); // 可以预约的座位为 [4,5] ,返回最小编号的座位,也就是 4 。
seatManager.reserve(); // 唯一可以预约的是座位 5 ,所以返回 5 。
seatManager.unreserve(5); // 将座位 5 变为可以预约,现在可预约的座位为 [5] 。
提示:
- 1 <= n <= 105
- 1 <= seatNumber <= n
- 每一次对 reserve 的调用,题目保证至少存在一个可以预约的座位。
- 每一次对 unreserve 的调用,题目保证 seatNumber 在调用函数前都是被预约状态。
- 对 reserve 和 unreserve 的调用 总共 不超过 105 次。
解题思路:
返回集合中的最小值,是堆的基本用法。
构建一个小顶堆,堆中的元素都是可以预约的,每次预约就从堆顶取出一个元素,然后返回即可。
取消预约,则将座位号重新加入堆中。
我的答案:
/**
* @param {number} n
*/
var SeatManager = function(n) {
this.can = new MinHeap();
for (let i=1; i<=n; i++) this.can.add(i);
};
/**
* @return {number}
*/
SeatManager.prototype.reserve = function() {
return this.can.pick();
};
/**
* @param {number} seatNumber
* @return {void}
*/
SeatManager.prototype.unreserve = function(seatNumber) {
this.can.add(seatNumber);
};
/**
* Your SeatManager object will be instantiated and called as such:
* var obj = new SeatManager(n)
* var param_1 = obj.reserve()
* obj.unreserve(seatNumber)
*/
class MinHeap {
constructor(arr = [], size = Infinity) {
this.heap = arr;
this.size = size;
for (let i = Math.floor(this.heap.length / 2) + 1; i >= 0; i--) {
this.down(i);
}
while (this.heap.length > this.size) this.heap.pop();
}
left(i) {
return 2 * i + 1;
}
right(i) {
return 2 * i + 2;
}
parent(i) {
return Math.floor((i - 1) / 2);
}
swap(i, j) {
[this.heap[i], this.heap[j]] = [this.heap[j], this.heap[i]];
}
down(i) {
let l = this.left(i);
let r = this.right(i);
let smaller = i;
if (l < this.heap.length && this.heap[l] < this.heap[smaller]) smaller = l;
if (r < this.heap.length && this.heap[r] < this.heap[smaller]) smaller = r;
if (smaller !== i) {
this.swap(smaller, i);
this.down(smaller);
}
}
up(i) {
let p = this.parent(i);
if (p >= 0 && this.heap[p] > this.heap[i]) {
this.swap(p, i);
this.up(p);
}
}
add(e) {
this.heap.push(e);
this.up(this.heap.length - 1);
while (this.heap.length > this.size) this.heap.pop();
}
top() {
return this.heap.length > 0 ? this.heap[0] : undefined;
}
getSize() {
return this.heap.length;
}
pick() {
this.swap(0, this.getSize()-1);
let r = this.heap.pop();
this.down(0);
return r;
}
}
最后
如果有更好的解法或者思路, 欢迎在评论区和我交流~ ღ( ´・ᴗ・` )