简体   繁体   English

boost :: tuple和std :: map的分段错误

[英]Segmentation Faults with boost::tuple and std::map

I have trouble with using code similar to the following one: 我在使用类似于以下代码的代码时遇到了麻烦:

std::map<boost::tuple<int, int, int>, int> m;
boost::tuple<int, int, int> key = boost::make_tuple(1,2,3);
m.find(key);

The compiler does not see any errors. 编译器看不到任何错误。 But when I start my program a weird Segmentation fault occurs. 但是,当我启动程序时,发生了奇怪的细分错误。 So I wanted to find the code line which causes it. 所以我想找到导致它的代码行。 GDB then told me: GDB然后告诉我:

Program received signal SIGSEGV, Segmentation fault.
0x0809f40a in boost::tuples::detail::lt<boost::tuples::cons<int,
boost::tuples::cons<int, boost::tuples::cons<int, boost::tuples::null_type> > >, 
boost::tuples::cons<int, boost::tuples::cons<int, boost::tuples::cons<int, 
boost::tuples::null_type> > > > (lhs=..., rhs=...)
at /usr/local/lib/boost_1_45_0/boost/tuple/tuple_comparison.hpp:73
73             lt(lhs.get_tail(), rhs.get_tail()));

Unfortunately I could not find any solutions to this kind of problem so far. 不幸的是,到目前为止,我找不到任何解决此类问题的方法。

Does anybody see what I missed here? 有人看到我在这里错过的东西吗?

EDIT: So I did some further investigation. 编辑:所以我做了一些进一步的调查。 The Object which causes the problem is a user-defined one. 引起问题的对象是用户定义的对象。 And actually neither the boost-stuff nor the usage of the map seems to be the reason because the error also occurs with vectors! 实际上,boost-stuff或映射的使用似乎都不是原因,因为矢量也会发生错误!

class A {
void foo();
private:
    std::vector<int> v;
}
void A::foo() {
    ...
    v = std::vector<int>(); // Here already comes a segfault.
    ...
}

I also tried to reproduce the error in a seperate class. 我还尝试在单独的类中重现该错误。 Unfortunately I wasn't able to provoke the error in there. 不幸的是,我无法引起那里的错误。

Now the gdb tells me: 现在,gdb告诉我:

Program received signal SIGSEGV, Segmentation fault.
0x010cae21 in free () from /lib/libc.so.6
(gdb) backtrace 
#0  0x010cae21 in free () from /lib/libc.so.6
#1  0x00fd7441 in operator delete(void*) () from /usr/lib/libstdc++.so.6
#2  0x080668c7 in __gnu_cxx::new_allocator<int>::deallocate (this=0xbfffdb04, 
__p=0x210bf) at /usr/include/c++/4.4/ext/new_allocator.h:95
#3  0x08064b8d in std::_Vector_base<int, std::allocator<int> >::_M_deallocate 
(this=0xbfffdb04, __p=0x210bf, __n=105047440)
at /usr/include/c++/4.4/bits/stl_vector.h:146
#4  0x0806246a in std::_Vector_base<int, std::allocator<int> >::~_Vector_base
(this=0xbfffdb04, __in_chrg=<value optimized out>)
at /usr/include/c++/4.4/bits/stl_vector.h:132
#5  0x080604d4 in std::vector<int, std::allocator<int> >::~vector (this=0xbfffdb04,  
__in_chrg=<value optimized out>)
at /usr/include/c++/4.4/bits/stl_vector.h:313
#6  0x0809e151 in ModelManager::emitSignal (this=0xbffff20f, o=crossroad, r=none, 
restrID=-1, signal=add, id=-5, colStart=-1, colEnd=-1)
at .build_debug/src/model/modelmanager.cpp:103

Could it be caused by the compiler settings? 可能是由编译器设置引起的吗?

#include <map>
#include <iostream>
#include <boost/tuple/tuple.hpp>
#include <boost/tuple/tuple_comparison.hpp>

int main()
{
    typedef boost::tuple<int, int, int> TTuple3;
    typedef std::map<TTuple3, int> TTupleMap;
    TTupleMap m;
    TTuple3 key1 = boost::make_tuple(1,2,3);
    TTuple3 key2 = boost::make_tuple(1,2,4);
    m[key1] = 1;
    m[key2] = 2;
    TTupleMap::iterator it = m.find(key1);
    if (it == m.end())
        std::cout << "not found" << std::endl;
    else
        std::cout << "found" << std::endl;

    std::cout << m[key1] << std::endl;
    std::cout << m[key2] << std::endl;

    return 0;
}

this produce for me: 这为我带来了:

found
1
2

Nothing wrong here. 没错

The behavior you're describing indicates you've corrupted your heap or stack. 您描述的行为表明您已损坏堆或堆栈。 I'd suggest recompiling with options -Wall -Wextra , fix any warnings you encounter and then see if you still run into the crash. 我建议使用选项-Wall -Wextra重新编译,修复遇到的所有警告,然后查看是否仍然崩溃。 I'd pay close attention to functions declared non-void that don't return a value. 我会密切注意声明为非void的不返回值的函数。 g++ doesn't warn by default for this and will almost always cause a crash away from where the error actually is. 默认情况下,g ++不会对此发出警告,并且几乎总是会导致崩溃,并远离错误的实际位置。 -Wextra will turn that warning on. -Wextra将打开该警告。

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM