- https://leetcode.cn/problems/longest-uploaded-prefix/
- 给你一个 n 个视频的上传序列,每个视频编号为 1 到 n 之间的 不同 数字,你需要依次将这些视频上传到服务器。请你实现一个数据结构,在上传的过程中计算 最长上传前缀 。
如果 闭区间 1 到 i 之间的视频全部都已经被上传到服务器,那么我们称 i 是上传前缀。最长上传前缀指的是符合定义的 i 中的 最大值 。
请你实现 LUPrefix 类:
LUPrefix(int n) 初始化一个 n 个视频的流对象。
void upload(int video) 上传 video 到服务器。
int longest() 返回上述定义的 最长上传前缀 的长度。
示例 1:
输入:
["LUPrefix", "upload", "longest", "upload", "longest", "upload", "longest"]
[[4], [3], [], [1], [], [2], []]
输出:
[null, null, 0, null, 1, null, 3]
解释:
LUPrefix server = new LUPrefix(4); // 初始化 4个视频的上传流
server.upload(3); // 上传视频 3 。
server.longest(); // 由于视频 1 还没有被上传,最长上传前缀是 0 。
server.upload(1); // 上传视频 1 。
server.longest(); // 前缀 [1] 是最长上传前缀,所以我们返回 1 。
server.upload(2); // 上传视频 2 。
server.longest(); // 前缀 [1,2,3] 是最长上传前缀,所以我们返回 3 。
提示:
1 <= n <= 105
1 <= video <= 105
video 中所有值 互不相同 。
upload 和 longest 总调用 次数至多不超过 2 * 105 次。
至少会调用 longest 一次。
并查集
-
i∼j视作一个包含了i到j节点的连通分量
-
longest查询的就是包含1的连通分量的大小
class LUPrefix {
vector<int> p;
int find(int x) {
return p[x] == x ? x : p[x] = find(p[x]);
}
public:
LUPrefix(int n) {
p.resize(n + 2);// 每个视频编号为 1 到 n,并且输入i后将其父节点设置为i+1的父节点,所以需要n+2的空间
for (int i = 1; i < n + 2; ++i) p[i] = i;
}
void upload(int x) {
p[x] = find(x + 1);//x的父节点设置为x+1的父节点,即merge x和x+1
}
int longest() {
return find(1) - 1;
}
};
/**
* Your LUPrefix object will be instantiated and called as such:
* LUPrefix* obj = new LUPrefix(n);
* obj->upload(video);
* int param_2 = obj->longest();
*/
集合set
class LUPrefix {
set<int> st;
public:
LUPrefix(int n) {
for (int i = 1; i <= n + 1; i++) {
st.insert(i);
}
}
void upload(int video) {
st.erase(video);
}
int longest() {
return *st.begin() - 1;
}
};