Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Is unique_ptr<Derived> to unique_ptr<Base> up-casting automatic?

I know it is possible that a derived class unique_ptr can take place where base class unique_ptr is required for polymorphic types. For example, while returning from function

unique_ptr<Base> someFunction()
{
     return make_unique<Derived>(new Derived());
}

or passing to function as argument.

// Function taking unique pointer
void someOtherFunction(unique_ptr<Base>&& ptr)
// Code calling this function
someOtherFunction(std::move(ptrToDerived));

My question is: Is this upcasting always automatic? Or do we need to explicitly perform it using dynamic_cast?

like image 541
Chadwick Robbert Avatar asked Apr 04 '16 11:04

Chadwick Robbert


1 Answers

The (draft) standard says:

// 20.8.1.2.1, constructors
...
template <class U, class E>
  unique_ptr(unique_ptr<U, E>&& u) noexcept;
template <class U>
  unique_ptr(auto_ptr<U>&& u) noexcept;

Those are constructors from any unique_ptr. The standard further restricts their usage by clauses like this:

24 Remarks: This constructor shall not participate in overload resolution unless U* is implicitly convertible to T* and D is the same type as default_delete<T>

The effect of this remark is that unique_ptr<T> is constructible from unique_ptr<U> precisely U* is convertible to T* (and all deleter requirements are met). In particular, when T is an unambiguous public base class of U.

Since the constructor is not explicit, it serves as an implicit converter from unique_ptr<U> to unique_ptr<T>.

like image 148
n. 1.8e9-where's-my-share m. Avatar answered Sep 24 '22 02:09

n. 1.8e9-where's-my-share m.