【发布时间】:2012-11-20 15:53:13
【问题描述】:
我有一些 C++ 代码失败并出现编译错误,除非它在 C++11 模式下运行,但我无法弄清楚为什么会出现这种情况,因为代码没有(明确)使用 C ++11 个特点:
#include <vector>
#include <map>
#include <boost/unordered_map.hpp>
struct SomeStruct {
boost::unordered_map<int, int> intMap;
};
int main(int argc, const char* argv[]) {
std::vector<SomeStruct> vals;
vals.resize(100);
}
在使用 gcc 4.6.3-1ubuntu5 和 boost 1.48 的 Ubuntu 12.04 64 位机器上编译时:
g++ test.cpp
然后我得到这个编译错误:
In file included from /usr/include/c++/4.6/vector:61:0,
from test.cpp:1:
/usr/include/c++/4.6/bits/stl_algobase.h: In function 'typename __gnu_cxx::__enable_if<(! std::__is_scalar<_Tp>::__value), void>::__type std::__fill_a(_ForwardIterator, _ForwardIterator, const _Tp&) [with _ForwardIterator = SomeStruct*, _Tp = SomeStruct, typename __gnu_cxx::__enable_if<(! std::__is_scalar<_Tp>::__value), void>::__type = void]':
/usr/include/c++/4.6/bits/stl_algobase.h:722:7: instantiated from 'void std::fill(_ForwardIterator, _ForwardIterator, const _Tp&) [with _ForwardIterator = SomeStruct*, _Tp = SomeStruct]'
/usr/include/c++/4.6/bits/vector.tcc:397:5: instantiated from 'void std::vector<_Tp, _Alloc>::_M_fill_insert(std::vector<_Tp, _Alloc>::iterator, std::vector<_Tp, _Alloc>::size_type, const value_type&) [with _Tp = SomeStruct, _Alloc = std::allocator<SomeStruct>, std::vector<_Tp, _Alloc>::iterator = __gnu_cxx::__normal_iterator<SomeStruct*, std::vector<SomeStruct> >, typename std::_Vector_base<_Tp, _Alloc>::_Tp_alloc_type::pointer = SomeStruct*, std::vector<_Tp, _Alloc>::size_type = long unsigned int, std::vector<_Tp, _Alloc>::value_type = SomeStruct]'
/usr/include/c++/4.6/bits/stl_vector.h:944:9: instantiated from 'void std::vector<_Tp, _Alloc>::insert(std::vector<_Tp, _Alloc>::iterator, std::vector<_Tp, _Alloc>::size_type, const value_type&) [with _Tp = SomeStruct, _Alloc = std::allocator<SomeStruct>, std::vector<_Tp, _Alloc>::iterator = __gnu_cxx::__normal_iterator<SomeStruct*, std::vector<SomeStruct> >, typename std::_Vector_base<_Tp, _Alloc>::_Tp_alloc_type::pointer = SomeStruct*, std::vector<_Tp, _Alloc>::size_type = long unsigned int, std::vector<_Tp, _Alloc>::value_type = SomeStruct]'
/usr/include/c++/4.6/bits/stl_vector.h:632:4: instantiated from 'void std::vector<_Tp, _Alloc>::resize(std::vector<_Tp, _Alloc>::size_type, std::vector<_Tp, _Alloc>::value_type) [with _Tp = SomeStruct, _Alloc = std::allocator<SomeStruct>, std::vector<_Tp, _Alloc>::size_type = long unsigned int, std::vector<_Tp, _Alloc>::value_type = SomeStruct]'
test.cpp:12:18: instantiated from here
/usr/include/c++/4.6/bits/stl_algobase.h:676:2: error: no match for 'operator=' in '* __first = __value'
/usr/include/c++/4.6/bits/stl_algobase.h:676:2: note: candidate is:
test.cpp:5:8: note: SomeStruct& SomeStruct::operator=(SomeStruct&)
test.cpp:5:8: note: no known conversion for argument 1 from 'const SomeStruct' to 'SomeStruct&'
在 gcc 中启用 C++11 支持时编译正常:
g++ -std=c++0x test.cpp
有人能解释一下为什么这只适用于 C++11 模式吗?
编辑:
使用 Vagrant 进行复制的步骤:
vagrant init precise64
vagrant up
vagrant ssh
sudo apt-get install -y build-essential libboost1.48-all-dev
echo "#include <vector>
#include <map>
#include <boost/unordered_map.hpp>
struct SomeStruct {
boost::unordered_map<int, int> intMap;
};
int main(int argc, const char* argv[]) {
std::vector<SomeStruct> vals;
vals.resize(100);
}" > test.cpp
g++ test.cpp
【问题讨论】:
-
我无法在 gcc 4.6.3 或 4.8、ubuntu 12.04 64 位和 boost 1.46 上重现此问题。
-
MinGW下使用gcc 4.7.0和boost 1.49没有错误,用
-std=c++03编译 -
对不起,这是 boost 1.48 而不是 1.46。使用 Vagrant 重现的步骤:
vagrant init precise64; vagrant up; vagrant ssh; sudo apt-get install -y build-essential libboost1.48-all-dev; echo "#include <vector> #include <map> #include <boost/unordered_map.hpp> struct SomeStruct { boost::unordered_map<int, int> intMap; }; int main(int argc, const char* argv[]) { std::vector<SomeStruct> vals; vals.resize(100); }" > test.cpp; g++ test.cpp -
尝试注释掉
std::vector::resize这一行。 -
@bamboon 代码编译良好,调整大小行被注释掉。