4

C++11 モードで実行しないとコンパイル エラーで失敗する C++ コードがいくつかありますが、コードが (明示的に) 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 およびブースト 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
4

1 に答える 1

3

のコピー代入演算子boost::unordered_mapは次のように宣言されます。

unordered_map& operator=(unordered_map &t)

これにより、(C ++ 98 12.8 [#10])の暗黙のコピー代入演算子が次のSomeStructように宣言されます。

SomeStruct &operator= (SomeStruct &);

明示的なコピー代入演算子の追加

SomeStruct &operator= (const SomeStruct &x) { intMap = x.intMap; return *this; }

問題を解決し、の別のオーバーロードを呼び出すことで機能しunordered_map::operator=ます。

この問題はBoost1.52で修正されています。これは、演算子を次のように宣言します。

unordered_map& operator=(unordered_map const& x);
于 2012-12-03T10:14:36.937 に答える