Векторное перераспределение использует копию вместо перемещения конструктора
Привет, я создал класс Foo с помощью noexcept move constructor с использованием gcc 4.7 и установил размер векторного резерва в 2, чтобы ему пришлось перераспределять размер при добавлении третьего элемента. Кажется, он вызывает конструктор копирования вместо конструктора перемещения при выполнении этого. Я что-то пропустил?
#include <vector>
#include <iostream>
class Foo
{
public:
Foo(int x) : data_(x)
{
std::cout << " constructing " << std::endl;
}
~Foo()
{
std::cout << " destructing " << std::endl;
}
Foo& operator=(const Foo&) = default;
Foo& operator=(Foo&&) = default;
Foo(Foo&& other) noexcept : data_(std::move(other.data_))
{
std::cout << " Move constructing " << std::endl;
}
Foo(const Foo& other) noexcept : data_(other.data_)
{
std::cout << " Copy constructing " << std::endl;
}
private:
int data_;
};
int main ( int argc, char *argv[])
{
std::vector<Foo> v;
v.reserve(2);
v.emplace_back(1);
std::cout << "Added 1" << std::endl;
v.emplace_back(2);
std::cout << "Added 2" << std::endl;
v.emplace_back(3);
std::cout << "Added 3" << std::endl;
std::cout << "v size: " << v.size() << std::endl;
}
выход:
constructing
Added 1
constructing
Added 2
constructing
Copy constructing
Copy constructing
destructing
destructing
Added 3
v size: 3
destructing
destructing
destructing
Ответы
Ответ 1
После небольшого перебора с GCC 4.7 и 4.8 кажется, что это действительно ошибка в 4.7, которая появляется только тогда, когда деструктор класса не помечен noexcept
:
struct Foo {
Foo() {}
~Foo() noexcept {}
Foo(Foo&&) noexcept { std::cout << "move constructor" << std::endl; }
Foo(const Foo&) noexcept { std::cout << "copy constructor" << std::endl; }
};
int main() {
std::vector<Foo> v;
v.reserve(2);
v.emplace_back();
v.emplace_back();
v.emplace_back();
}
GCC 4.7 отображает:
move constructor
move constructor
Если мы удалим noexcept
из деструктора:
struct Foo {
Foo() {}
~Foo() {}
Foo(Foo&&) noexcept { std::cout << "move constructor" << std::endl; }
Foo(const Foo&) noexcept { std::cout << "copy constructor" << std::endl; }
};
GCC 4.7 отображает:
copy constructor
copy constructor
GCC 4.8 использует конструктор перемещения в обоих случаях.