news 2026/6/10 2:28:46

二分查找(十)1146. 快照数组 pair整理

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
二分查找(十)1146. 快照数组 pair整理

1146. 快照数组

实现支持下列接口的「快照数组」- SnapshotArray:

  • SnapshotArray(int length)- 初始化一个与指定长度相等的 类数组 的数据结构。初始时,每个元素都等于0
  • void set(index, val)- 会将指定索引index处的元素设置为val
  • int snap()- 获取该数组的快照,并返回快照的编号snap_id(快照号是调用snap()的总次数减去1)。
  • int get(index, snap_id)- 根据指定的snap_id选择快照,并返回该快照指定索引index的值。

示例:

输入:["SnapshotArray","set","snap","set","get"] [[3],[0,5],[],[0,6],[0,0]]输出:[null,null,0,null,5]解释:SnapshotArray snapshotArr = new SnapshotArray(3); // 初始化一个长度为 3 的快照数组 snapshotArr.set(0,5); // 令 array[0] = 5 snapshotArr.snap(); // 获取快照,返回 snap_id = 0 snapshotArr.set(0,6); snapshotArr.get(0,0); // 获取 snap_id = 0 的快照中 array[0] 的值,返回 5

解法一:使用map存储数组的情况虽然方便理解,但是遇到大量请求的时候,导致大量的深拷贝操作进行,超时。

class SnapshotArray { public: SnapshotArray(int length) { ShotArray.resize(length, 0); } void set(int index, int val) { ShotArray[index] = val; } int snap() { ShotArrays[snap_id++] = ShotArray; return snap_id-1; } int get(int index, int snap_id) { return ShotArrays[snap_id][index]; } private: map<int, vector<int>> ShotArrays; vector<int> ShotArray; int snap_id = 0; };

思路
假设每调用一次 set,就生成一个快照(复制一份数组)。仅仅是一个元素发生变化,就去复制整个数组,这太浪费了。

能否不复制数组呢?

换个视角,调用 set(index,val) 时,不去修改数组,而是往 index 的历史修改记录末尾添加一条数据:此时的快照编号和 val。

举例说明:

在快照编号等于 2 时,调用 set(0,6)。
在快照编号等于 3 时,调用 set(0,1)。
在快照编号等于 3 时,调用 set(0,7)。
在快照编号等于 5 时,调用 set(0,2)。
这四次调用结束后,下标 0 的历史修改记录 history[0]=[(2,6),(3,1),(3,7),(5,2)],每个数对中的第一个数为调用 set 时的快照编号,第二个数为调用 set 时传入的 val。注意历史修改记录中的快照编号是有序的。
那么:

调用 get(0,4)。由于历史修改记录中的快照编号是有序的,我们可以在 history[0] 中二分查找快照编号 ≤4 的最后一条修改记录,即 (3,7)。修改记录中的 val=7 就是答案。
调用 get(0,1)。在 history[0] 中,快照编号 ≤1 的记录不存在,说明在快照编号 ≤1 时,我们没有修改下标 0 保存的元素,返回初始值 0。

class SnapshotArray { // 建议:history 改名为 records 可能更直观 unordered_map<int, vector<pair<int,int>>> history; int snap_id = 0; public: SnapshotArray(int length) { // map 不需要预分配空间,这里空着也没事 } void set(int index, int val) { // 【修正1】加上引用 &,或者直接操作 map // 这样才能真正修改 map 里的数据 history[index].push_back({snap_id, val}); } int snap() { snap_id++; return snap_id - 1; } int get(int index, int snap_id) { // 【注意】这里不要用 auto it = history[index],因为会产生巨大的拷贝开销! // 如果只是读取,最好用引用,或者直接用 history.find(index) // 如果这个 index 从来没存过数据,直接返回 0 if (history.find(index) == history.end()) { return 0; } auto& vec = history[index]; // 加上引用 & 避免拷贝!! // 二分查找 auto it = upper_bound(vec.begin(), vec.end(), make_pair(snap_id, 2000000000)); // 【修正2】判断边界 // 如果它是 begin(),说明所有记录的 snap_id 都比查询的 snap_id 大(或者数组为空) // 这种情况下应该返回初始值 0 if (it == vec.begin()) { return 0; } // 安全地回退一步 return prev(it)->second; } };

使用auto拿一个对象里面数据的时候,不管要不要进行修改,都尽量加上引用;

lower_bound找不到返回end() upper_bound找不到返回begin()

#include <iostream> #include <utility> #include <vector> #include <algorithm> using namespace std; int main() { // 1. 增 pair<int, int> p = {5, 10}; // 2. 改 p.first = 99; // 3. 查 (C++17 酷炫写法) auto [x, y] = p; cout << "x: " << x << ", y: " << y << endl; // x: 99, y: 10 // 4. 排序演示 vector<pair<int, int>> vec = {{2, 10}, {1, 20}, {1, 5}}; sort(vec.begin(), vec.end()); // 排序后顺序:{1, 5}, {1, 20}, {2, 10} for(auto& item : vec) { cout << "{" << item.first << "," << item.second << "} "; } return 0; }

lambda自定义排序

sort(vec.begin(), vec.end(), [](const auto& a, const auto& b) { // 逻辑:return true 代表 a 应该排在 b 前面 // 情况1:主关键字不同,按主关键字排(比如按 value 降序) if (a.second != b.second) { return a.second > b.second; } // 情况2:主关键字相同,按次关键字排(比如按 key 升序) return a.first < b.first; });
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/9 23:59:40

计算机Java毕设实战-基于Web的旅游社交分享系统设计与实现基于Web的社交媒体平台基于web的宠物社交系统【完整源码+LW+部署说明+演示视频,全bao一条龙等】

博主介绍&#xff1a;✌️码农一枚 &#xff0c;专注于大学生项目实战开发、讲解和毕业&#x1f6a2;文撰写修改等。全栈领域优质创作者&#xff0c;博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战 ✌️技术范围&#xff1a;&am…

作者头像 李华
网站建设 2026/6/10 14:11:26

奥特曼被吓坏!Codex全家桶上线倒计时,恐将撕开全网漏洞

奥特曼发出预警&#xff1a;一周后Codex全家桶就要来了&#xff0c;但它们极其危险&#xff0c;以至于网络安全评级已经到达高级别&#xff01;这些模型极可能打破现有的网络攻防平衡&#xff0c;导致攻击数量激增&#xff0c;甚至能帮你抢银行。 今天&#xff0c;奥特曼预告&…

作者头像 李华
网站建设 2026/6/10 15:44:57

代码死了,死在Cursor生成3000000行浏览器的那个晚上!

这是硅谷近期最大的黑色幽默&#xff1a;Cursor CEO亲拆自家招牌。当「Vibe Coding」变成一场盲目的狂欢&#xff0c;技术次贷危机的引信已被悄悄点燃。 就在上周&#xff0c;科技圈围观了一场名为奇迹&#xff0c;实为葬礼的闹剧。 Cursor的CEO Michael Truell&#xff0c;这…

作者头像 李华
网站建设 2026/6/10 16:02:54

全域风控与反欺诈技术实践——直击刷单、黄牛、恶意下单及盗刷痛点

【精选优质专栏推荐】 《AI 技术前沿》 —— 紧跟 AI 最新趋势与应用《网络安全新手快速入门(附漏洞挖掘案例)》 —— 零基础安全入门必看《BurpSuite 入门教程(附实战图文)》 —— 渗透测试必备工具详解《网安渗透工具使用教程(全)》 —— 一站式工具手册《CTF 新手入门实战教…

作者头像 李华
网站建设 2026/6/9 23:42:17

Sentinel 技术原理及通信端口说明

当启动 Java sentinel客户端应用时&#xff0c;除了监听 8080 端口提供业务接口外&#xff0c;还会监听 8719 端口用于与 Sentinel Dashboard 通信。 端口用途说明&#xff1a; 8080 端口&#xff1a; 应用的主服务端口用于接收和处理业务请求在本项目中对应 /api/test 等接口…

作者头像 李华