- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在跟踪一个错误,但遇到了非常奇怪的行为。我有一组指针,当我一个一个地删除它们时,第一个删除,但删除另一个给我段错误。我用
size_type erase( const key_type& key );
所以它不可能是迭代器。我的调试器在调用堆栈中向我显示:
0 - std::less<cSubscriber *>::operator() //cSubscriber is an abstract base class and I have a set of cSubscriber *
1 - std::_Rb_tree<cSubscriber*, cSubscriber*, std::_Identity<cSubscriber*>, std::less<cSubscriber*>, std::allocator<cSubscriber*> >::equal_range
2 - std::_Rb_tree<cSubscriber*, cSubscriber*, std::_Identity<cSubscriber*>, std::less<cSubscriber*>, std::allocator<cSubscriber*> >::erase
3 - std::set<cSubscriber*, std::less<cSubscriber*>, std::allocator<cSubscriber*> >::erase
4 - cEventSystem::unsubscribe //my function, it is as follows in the class which has the set as its member
cEventSystem::unsubscribe(cSubscriber * ptr)
{
set.erase(ptr);
}
并且在基类 cSubscriber 抽象类中有虚析构函数:
virtual ~cSubscriber()
{
eventSystem.unsubscribe(this);
}
有什么想法吗?我不知道它怎么会导致段错误,当没有这样的元素时,删除应该只返回 0。或者在尝试从空容器中删除某些东西时可能会崩溃? (我有另一个错误,当添加 3 个不同的指针后,集合的大小只有 2,但那是另一回事了)。
最佳答案
如果您将无效地址传递给您的 std::set<SOMETHING*>::erase()
尝试将传递的值与容器中的值进行比较时会出现段错误。
例如:
struct IntPtrComparer {
bool operator()(int* a, int* b) const {
return *a < *b;
}
};
std::set<int*,IntPtrComparer> a;
a.insert(new int);
a.erase(NULL);
基于评论
由于您没有重新定义默认比较器,并且默认比较器不会取消引用您的指针,那么唯一的方法就是您的 std::set
已损坏。
在内部,std::set
被实现为二叉树。这意味着它在查找值和删除值的过程中有很多指针。如果std::set
这些指针中的一些已损坏将指向无效的内存地址。这个无效的内存地址将用于将比较值的引用(指针的引用,cSubscriber* & const
)传递给std::less
。 . std::less
接收到一个指针的引用,并将取消对该引用的引用以获取指针值。
这样,std::set
中的无效内存只出现在std::less
,因为 std::set
实际上并没有触及无效内存,它把无效内存地址给了我们可怜的家伙std::less
打开它并在它的脸上出现段错误。
我的观点是,如果您创建一个使用拷贝而不是引用的比较器,损坏将出现在 std::set
中。当它试图复制指针值以提供给比较器时。
关于c++ - std::set::erase(const key_type& key) 如何导致段错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14736002/
假设您要按值从 vector 中删除单个元素。 remove 之间有什么区别? -删除: vector v; // add some values vector::iterator it = remo
class CSensor { public: CSensor(int nVal1,char* pVal2,unsigned int nVal3); CSensor(cons
在回答过程中another question我偶然发现 std::vector::erase() 和 std::deque::erase() 的措辞略有不同。 这就是 C++14 关于 std::de
我想处理 vector 中的元素一段时间。为了优化这一点,我不想在处理项目时删除它,而是在最后删除所有已处理的项目。 vector::iterator it; for(it = items.begin
是否 std::set erase ( x ) 在 STL 中以相同的方式实现 erase ( collection.find( x ) ); (其中 x 是 const key_type&) 吗?
This question already has answers here: How to catch .erase exception in vectors? (2个答案) 8天前关闭。 当我尝试
我有以下代码: #include using namespace std; int main() { set S; S.insert("item1"); S.insert("i
在'helper'函数中输出集合元素时,预计元素'2'已经被移除。但实际结果仍然是'1 2 3 4 5'。 但是,集合的大小是 4。 我想知道潜在的问题。 #include #include us
有点难以解释,但我想制作一个以图片为背景的网站,并在图片上覆盖一层灰色。然后我想在覆盖层上方添加一些文本,在文本所在的位置,灰色覆盖层被移除,因此文本是在没有灰色覆盖层的情况下写入的。 我在一年前用图
这是导致错误的代码: 工厂.h: #include #include namespace BaseSubsystems { template class CFactory
我正在研究结构 vector 。 当我试图用迭代器调用这个函数时,像这样: vec2.erase (vec2.begin()+iter2); 它向我发送了这个错误: "no match for 'op
进程文件: eraser or eraser.exe 进程名称: Eraser 进程类别:存在安全风险的进程 英文描述: eraser.exe is the main executable
我正在编写一个简单的程序,它使用了 std::map::erase . 该程序很好,但有一些我不明白的地方。 如果我传给 erase function 第一个迭代器超出第二个迭代器的间隔,该函数不会删
如果我的变量是 0 到 6294 之间的随机数,我如何让我的输出仅打印出第一个变量? 我的随机数代码是 int random = (int)(Math.random() * 6294); 例如,如果它
我正在尝试通过一个测试程序,并且我通过了所有测试,除了涉及到我的删除功能时,程序崩溃了。 我最好的猜测可能是前面或后面的哨兵节点正在被删除。或内存泄漏。 最佳答案 我怀疑有人愿意阅读所有这些代码并为您
背景:这是一个程序,用于将存储在 vector 中的数字逐位加 1。 前面有 0 的数字可以作为输入,但不能作为输出。例如:0123 和 123 都是有效输入 但 0124 是无效输出 而 124 是
我正在尝试创建一个垂直滚动的射击游戏,当您按下空格键时,会创建一颗子弹,然后当子弹离开屏幕时,子弹就会被销毁。我通过声明为 vector bullets; 的 vector 跟踪子弹当我试图销毁屏幕外
我在下面写这段代码,发现了这个奇怪的行为: #include #include #include using namespace std; int main() { map map1;
我为 SFML 编写了一个线程渲染器,它接受指向可绘制对象的指针并将它们存储在一个 vector 中以在每一帧中绘制。开始向 vector 添加对象和从 vector 中删除对象会经常导致段错误 (S
如果我想从 map 中删除单个元素(除了可能的错误检查之外我不关心返回值),有两种方法可以实现:erase按值键或删除: http://ideone.com/YWocN7 #include #inc
我是一名优秀的程序员,十分优秀!