请你设计一个管理 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 <= 10^5
1 <= seatNumber <= n
- 每一次对
reserve
的调用,题目保证至少存在一个可以预约的座位。 - 每一次对
unreserve
的调用,题目保证seatNumber
在调用函数前都是被预约状态。 - 对
reserve
和unreserve
的调用 总共 不超过10^5
次。
提示 1
You need a data structure that maintains the states of the seats. This data structure should also allow you to get the first available seat and flip the state of a seat in a reasonable time.
提示 2
You can let the data structure contain the available seats. Then you want to be able to get the lowest element and erase an element, in a reasonable time.
提示 3
Ordered sets support these operations.
解法1:最小堆(优先队列)
我们使用数据结构 q 来维护所有可以预约的座位,我们需要分析 reserve 与 unreserve 的具体需求:
- 对于 reserve 方法,我们需要弹出并返回 q 中的最小元素;
- 对于 unreserve 方法,我们需要将 seatNumber 添加至 q 中。
因此我们可以使用二叉堆实现的优先队列作为 q。对于一个最小堆,可以在 O(logn) 的时间复杂度内完成单次「添加元素」与「弹出最小值」的操作。
Java版:
class SeatManager {
PriorityQueue<Integer> q;
public SeatManager(int n) {
q = new PriorityQueue<>();
for (int i = 1; i <= n; i++) {
q.offer(i);
}
}
public int reserve() {
return q.poll();
}
public void unreserve(int seatNumber) {
q.offer(seatNumber);
}
}
/**
* Your SeatManager object will be instantiated and called as such:
* SeatManager obj = new SeatManager(n);
* int param_1 = obj.reserve();
* obj.unreserve(seatNumber);
*/
Python3版:
class SeatManager:
def __init__(self, n: int):
self.q = [i for i in range(1, n + 1)]
heapify(self.q)
def reserve(self) -> int:
return heappop(self.q)
def unreserve(self, seatNumber: int) -> None:
heappush(self.q, seatNumber)
# Your SeatManager object will be instantiated and called as such:
# obj = SeatManager(n)
# param_1 = obj.reserve()
# obj.unreserve(seatNumber)
复杂度分析
- 时间复杂度:O(n+(q1+q2)logn),其中 n 为座位的数量,q1 为 reserve 操作的次数,q2 为 unreserve 的次数。初始化的时间复杂度为 O(n),二叉堆实现的优先队列单次添加元素与弹出最小值操作的复杂度均为 O(logn)。
- 空间复杂度:O(n),二叉堆的空间开销。