Почему приведенный ниже код не компилируется с gcc, но прекрасно компилируется с помощью clang
Ниже код хорошо компилируется с помощью clang, но не с помощью gcc. Любое объяснение - это ошибка в gcc?
Это просто класс, который содержит вектор unique_ptr и std::function в качестве члена, и когда я создаю вектор этого класса, я не могу сказать зарезервировать или изменить его размер. push_back отлично работает с std::move, в то время как это происходит только с gcc, а не clang.
#include <algorithm>
#include <memory>
#include <utility>
#include <iostream>
#include <vector>
#include <functional>
using namespace std;
class ABC
{
public:
ABC()
{}
private:
std::vector<std::unique_ptr<int>> up;
std::function<void (int*)> func;
};
int main()
{
ABC a;
std::vector<ABC> vec;
vec.reserve(1);
}
Сообщение об ошибке выглядит как ниже для GCC
In file included from /opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/stl_tempbuf.h:60:0,
from /opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/stl_algo.h:62,
from /opt/wandbox/gcc-7.1.0/include/c++/7.1.0/algorithm:62,
from prog.cc:1:
/opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/stl_construct.h: In instantiation of 'void std::_Construct(_T1*, _Args&& ...) [with _T1 = std::unique_ptr<int>; _Args = {const std::unique_ptr<int, std::default_delete<int> >&}]':
/opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/stl_uninitialized.h:83:18: required from 'static _ForwardIterator std::__uninitialized_copy<_TrivialValueTypes>::__uninit_copy(_InputIterator, _InputIterator, _ForwardIterator) [with _InputIterator = __gnu_cxx::__normal_iterator<const std::unique_ptr<int>*, std::vector<std::unique_ptr<int> > >; _ForwardIterator = std::unique_ptr<int>*; bool _TrivialValueTypes = false]'
/opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/stl_uninitialized.h:134:15: required from '_ForwardIterator std::uninitialized_copy(_InputIterator, _InputIterator, _ForwardIterator) [with _InputIterator = __gnu_cxx::__normal_iterator<const std::unique_ptr<int>*, std::vector<std::unique_ptr<int> > >; _ForwardIterator = std::unique_ptr<int>*]'
/opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/stl_uninitialized.h:289:37: required from '_ForwardIterator std::__uninitialized_copy_a(_InputIterator, _InputIterator, _ForwardIterator, std::allocator<_Tp>&) [with _InputIterator = __gnu_cxx::__normal_iterator<const std::unique_ptr<int>*, std::vector<std::unique_ptr<int> > >; _ForwardIterator = std::unique_ptr<int>*; _Tp = std::unique_ptr<int>]'
/opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/stl_vector.h:331:31: required from 'std::vector<_Tp, _Alloc>::vector(const std::vector<_Tp, _Alloc>&) [with _Tp = std::unique_ptr<int>; _Alloc = std::allocator<std::unique_ptr<int> >]'
prog.cc:10:7: required from 'void std::_Construct(_T1*, _Args&& ...) [with _T1 = ABC; _Args = {const ABC&}]'
/opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/stl_uninitialized.h:83:18: required from 'static _ForwardIterator std::__uninitialized_copy<_TrivialValueTypes>::__uninit_copy(_InputIterator, _InputIterator, _ForwardIterator) [with _InputIterator = const ABC*; _ForwardIterator = ABC*; bool _TrivialValueTypes = false]'
/opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/stl_uninitialized.h:134:15: required from '_ForwardIterator std::uninitialized_copy(_InputIterator, _InputIterator, _ForwardIterator) [with _InputIterator = const ABC*; _ForwardIterator = ABC*]'
/opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/stl_uninitialized.h:289:37: required from '_ForwardIterator std::__uninitialized_copy_a(_InputIterator, _InputIterator, _ForwardIterator, std::allocator<_Tp>&) [with _InputIterator = const ABC*; _ForwardIterator = ABC*; _Tp = ABC]'
/opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/stl_vector.h:1263:35: required from 'std::vector<_Tp, _Alloc>::pointer std::vector<_Tp, _Alloc>::_M_allocate_and_copy(std::vector<_Tp, _Alloc>::size_type, _ForwardIterator, _ForwardIterator) [with _ForwardIterator = const ABC*; _Tp = ABC; _Alloc = std::allocator<ABC>; std::vector<_Tp, _Alloc>::pointer = ABC*; std::vector<_Tp, _Alloc>::size_type = long unsigned int]'
/opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/vector.tcc:73:40: required from 'void std::vector<_Tp, _Alloc>::reserve(std::vector<_Tp, _Alloc>::size_type) [with _Tp = ABC; _Alloc = std::allocator<ABC>; std::vector<_Tp, _Alloc>::size_type = long unsigned int]'
prog.cc:24:18: required from here
/opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/stl_construct.h:75:7: error: use of deleted function 'std::unique_ptr<_Tp, _Dp>::unique_ptr(const std::unique_ptr<_Tp, _Dp>&) [with _Tp = int; _Dp = std::default_delete<int>]'
{ ::new(static_cast<void*>(__p)) _T1(std::forward<_Args>(__args)...); }
^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
In file included from /opt/wandbox/gcc-7.1.0/include/c++/7.1.0/memory:80:0,
from prog.cc:2:
/opt/wandbox/gcc-7.1.0/include/c++/7.1.0/bits/unique_ptr.h:388:7: note: declared here
unique_ptr(const unique_ptr&) = delete;
^~~~~~~~~~
3 ответа
Это происходит потому, что std::function
ход ctor не noexcept
, но std::vector
можно использовать только ход ctor, если это noexcept
(гарантия сильного исключения).
Проблема в том, что std::unique_ptr
(очевидно) не копируемый, так что делает ABC
не копируемый в целом. Делать ABC
noexcept
-перемещаемый неявно, он должен был бы каждый из его членов быть noexcept
-движной, а также.
Если вы удалите std::function
вот что происходит: .resize()
не нужно копировать A.up
(а std::vector
ход оператора noexcept
), Итак std::unique_ptr
s (внутри up
) можно просто переместить, и все работает отлично.
Смотрите это колиру: если вы прокомментируете noexcept
, vec.reserve()
нужно будет все скопировать, и проблема вернется.
Добавление ABC(ABC&&) = default
решает проблему, потому что пользователь объявил (хотя default
ed) move ctor запрещает генерацию конструктора копирования (см. здесь).
Мы также можем пойти противоположным путем и вручную удалить A
Копируй конструктор в колиру (и держи ход не noexcept
): здесь.
Чтобы решить эту проблему, вы можете сохранить std::function
в std::unique_ptr
, который имеет nothrow
переместить ctor. Посмотреть здесь
Я думаю, что GCC требователен к конструкторам; блокирует генерацию move-ctor1. Он перестает жаловаться после того, как вы предоставите их явно:
class ABC
{
public:
ABC() = default; // <--
ABC(ABC&&) = default; // <--
private:
std::vector<std::unique_ptr<int>> up;
std::function<void (int*)> func;
};
1 Я думаю (опять же, я не уверен), что это происходит потому, что по умолчанию copy-ctor of ABC
определяется как noexcept
и как таковой предпочтительнее по умолчанию move-ctor (он все еще плохо сформирован, потому что мы имеем дело с не копируемыми членами здесь). Попытка создать дефолт noexcept
Результат Move-Ctor в:
main.cpp:14:4: note: 'ABC::ABC(ABC&&) noexcept' is implicitly deleted because its exception-specification does not match the implicit exception-specification ''
ABC(ABC&&) noexcept = default;
^~~
Таким образом, заставляя поколение неnoexcept
Move-Ctor позволяет его выбрать. Почему у Кланга нет проблем с этим - я не знаю.
Одно из объяснений вышесказанного намекает на то, что удаление std::function
позволяет дефолт noexcept
Move-Ctor будет создан. std::function
не имеет noexcept
move-ctor (который ужасно отстой), поэтому весь класс возвращается к copy-ctor. поскольку unique_ptr
не один, все ломается.
GCC признал отсутствие noexcept для ctor-перемещения std::function как ошибку
https://gcc.gnu.org/bugzilla/show_bug.cgi?id=81017
// -- C++ --
/** 2227 * @brief %Function move constructor. 2228 * @param
__x A %function object rvalue with identical call signature. 2229 * 2230
* The newly-created %function contains the target of @a __x 2231 * (if
it has one). 2232 */ 2233
function(function&& __x) : _Function_base() { __x.swap(*this); }