我有一个关于 std::set 容器的简短问题。现在我正在使用推回功能喂我的套装。当然,每个 push_back 的集合都会变得越来越大。我只对最新的30个左右的元素感兴趣...较旧的元素可以删除。所以我的想法是将集合的大小限制为 30 个左右的元素,并通过这样做摆脱不需要的旧元素。但是,该集合默认不支持限制。我可以不时检查集合的大小并手动删除多余的元素。有没有更聪明的方法?
问候隆皮
作为一种解决方案,您可以将set
数据结构封装到一个类中,并在该类中控制元素的数量。
您需要自己构建 LRU 结构。一种方法是让 std::map 和 std::list 指向彼此的迭代器。那是:
struct lru_entry {
std::list<lru_entry *>::iterator lru_iterator;
std::map<your_data, lru_entry *>::iterator map_iterator;
};
std::list<lru_entry *> lru;
std::map<your_data, lru_entry *> lookup;
每当您在地图中查找条目时,将其关联的列表条目移动到列表的开头。向映射添加条目时,创建一个新的 lru_entry,将其添加到映射和列表中,并更新 lru_entry 结构中的迭代器。当查找图超过 30 个条目时,您可以使用 lru 列表快速找到最旧的条目。
您可以在上一个 stackoverflow 问题中找到有关如何构建 LRU 列表的更多建议。
该集合不仅不支持限制,它也不告诉您元素的年龄。创建一个封装容器的新类。然后,该类可以提供方法来在添加元素时或显式地强制执行大小限制。如果根据添加元素的时间完成删除(它针对两端的操作进行了优化),则队列将是理想的。
由于我有时间,我会使用一组和一个列表来完成。该列表仅跟踪最后 n 个插入的元素。还实现了一般擦除。为了获得更好的性能(如果您没有利用 set 是有序的这一事实),您可以考虑使用 unordered_set。(替换include <set>
为include <unordered_set>
和)std::set
std::unordered_set
#include <set>
#include <list>
#include <assert.h>
template<typename T>
struct setkeepn {
std::set<T> mSet;
std::list<T> mLast;
void insert(T element) {
if (mSet.find(element) == mSet.end()) {
assert(mSet.size() == mLast.size());
// put your limit of 30 below
if (mSet.size() >= 2) {
T extra = mLast.back();
mSet.erase(extra);
mLast.pop_back();
}
mSet.insert(element);
mLast.push_front(element);
}
}
void erase(T element)
{
typename std::set<T>::iterator lToEraseFromSet = mSet.find(element);
if (lToEraseFromSet != mSet.end()) {
// linear on the number of elements.
typename std::list<T>::iterator lToEraseFromList = std::find(mLast.begin(),mLast.end(), element);
assert(lToEraseFromList != mLast.end());
mSet.erase(lToEraseFromSet);
mLast.erase(lToEraseFromList);
}
}
};
int main(int argc, const char * argv[]) {
setkeepn<int> lTest;
lTest.insert(1);
lTest.insert(2);
lTest.insert(2);
lTest.insert(3);
printf("should see 2 3\n");
for (auto e : lTest.mSet) { // 2,3
printf("elements: %d\n",e);
}
lTest.insert(4);
lTest.erase(3);
printf("should see 4\n");
for (auto e : lTest.mSet) { // 4
printf("elements: %d\n",e);
}
return 0;
}
结果:
should see 2 3
elements: 2
elements: 3
should see 4
elements: 4