c++ - 如何根据 .first 值从 std::pair 的 std::vector 中删除元素?
问题描述
好的,所以我有一个排序的std::vector<std::pair<int,double>>
. 我似乎无法找到的是如何根据 std::pair (int) 的“第一个”元素的值从向量中删除一个条目。我可能会在我的算法中多次这样做,所以我不想每次都遍历向量(可能包含多达一百万个条目)。我知道我们可以使用 std::erase 或 remove 轻松地根据索引删除元素,但是有没有办法根据对的第一个元素的值来做到这一点?或者我们可以得到那个元素的索引然后使用 std::erase 吗?
注意: std::pair 的第一个元素的值对于向量是唯一的。鉴于程序的限制,我需要使用向量(即不能使用地图或不同的容器)。
示例:我有一个这样的容器:
std::vector<std::pair<int,double>> vec = { {20, 60.3}, ... {10, -20.2}, {1020, -80.9}};
我想快速从向量中删除第一个元素 == 10 的元素,但我不知道它位于向量的哪个索引处。
解决方案
您的向量已排序,因此您可以(并且应该)使用std::lower_bound
and std::upper_bound
。
这些为您提供了与某些标准匹配的范围(前提是容器的排序顺序使其有意义),并通过良好的二进制搜索来实现。
提供一个只检查每对的第一项的自定义比较器。
#include <utility>
#include <vector>
#include <algorithm>
int main()
{
std::vector<std::pair<int,double>> data = { {20, 60.3}, {10, -20.2}, {1020, -80.9}};
const int intToSearchFor = 10;
const auto lower = std::lower_bound(
data.begin(),
data.end(),
intToSearchFor,
[](const std::pair<int, double>& el, const int i)
{
return el.first < i;
}
);
const auto upper = std::upper_bound(
data.begin(),
data.end(),
intToSearchFor,
[](const int i, const std::pair<int, double>& el)
{
return i < el.first;
}
);
data.erase(lower, upper);
}
如果您int
的 s 是唯一的,则不需要上限检查,并且可以擦除位置处的元素lower
……但是您必须首先确保它实际上等于i
(可能大于),而且它不是data.end()
。
该算法基本上实现std::map::erase
(或std::multimap::erase
)但在连续存储中使用排序数据。它非常适合快速查找相对较小的数据集;不幸的是,你被一个擦除后的后续元素洗牌的成本所困扰。地图通过间接存储数据来避免这种情况。双端队列对你来说可能是一个很好的中间立场。根据您的正常数据和访问模式,只有您可以知道。
您可能还会发现,因为元素类型只是 a pair<int, double>
,您的编译器可能会将整个operator=
调用负载换成一个不错的 simple memmove
,这在您现在谈论的规模上已经足够快了。
推荐阅读
- android - 如何使用 Firebase 将毕加索图像添加到 ArrayList
- xslt - 创建 html 表时避免重复
- nativescript - 当应用程序在 android 平台的前台时,Firebase 推送通知不显示
- php - Symfony:在 FormType 表单中为 EntityType 使用不同的字段
- selenium - 在命令提示符下运行 selenium 测试套件时有等待/超时
- javascript - 在 react 和 JavaScript 中获取正确格式的数据
- javascript - 在 MS Edge 中检测覆盖模式
- julia - 在 Julia/DifferentialEquations 中使用 VectorContinuousCallback 时输出解中事件的索引
- .net - Zip 文件的文件类型在复制到 AWS S3 时发生变化
- android - 使用 adb 和活动管理器在 Android 上捕获具有指定名称的 JPEG 图像