自C ++中的GCC 6以来,unique_ptr<T[]>::reset
方法的声明/定义(不是仅接受nullptr_t
的方法)如下所示:
template <typename _Up,
typename = _Require<
__or_<is_same<_Up, pointer>,
__and_<is_same<pointer, element_type*>,
is_pointer<_Up>,
is_convertible<
typename remove_pointer<_Up>::type(*)[],
element_type(*)[]
>
>
>
>>
void
reset(_Up __p) noexcept
{
using std::swap;
swap(std::get<0>(_M_t), __p);
if (__p != nullptr)
get_deleter()(__p);
}
在某些时候改变了这一点以实现N4089。根据该文件:
此函数的行为与主模板的重置成员相同,但它除外 除非
,否则不得参与重载决议-
的类型相同U
与pointer
或-
pointer
与element_type*
的类型相同,U
是指针类型V*
,V(*)[]
可转换为element_type(*)[]
。
让我们考虑以下示例:
std::unique_ptr<const char []> ptr1;
std::unique_ptr<char []> ptr2(new char[5]);
ptr1 = std::move(ptr2);
由于版本6 GCC产生错误,因此抱怨它无法使用std::swap
和const char*&
来呼叫char*&
。
reset
方法在重载决策中发生,char[]
可转换为const char[]
,但自然std::swap
等待两个相同类型的引用。
这被认为是正确的行为吗?如果是这样,为什么呢?如果我可以隐式地将char[]
转换为const char[]
,为什么unique_ptr
无法实现相同的目标?