web-dev-qa-db-ja.com

C ++ 11でmake_unique関数を実装する方法は?

私のコンパイラはmake_uniqueをサポートしていません。書き方は?

template< class T, class... Args > unique_ptr<T> make_unique( Args&&... args );
26
user1899020

make_unique and perfect forwarding からコピー(同じことが Herb Sutterのブログ に記載されています)

template<typename T, typename... Args>
std::unique_ptr<T> make_unique(Args&&... args)
{
    return std::unique_ptr<T>(new T(std::forward<Args>(args)...));
}

VC2012で必要な場合は、 VS2012でmake_unique()を記述する方法はありますか? を参照してください。


それでも、 sasha.sochkaの答え の解決策があなたのコンパイラでコンパイルされるなら、私はそれを使います。これはより複雑で、配列でも機能します。

30
Ali

C++ 14にこの関数を追加することを最初に提案したStephan T. Lavavej(STLとも呼ばれる)によるバージョン

#include <cstddef>
#include <memory>
#include <type_traits>
#include <utility>

namespace std {
    template<class T> struct _Unique_if {
        typedef unique_ptr<T> _Single_object;
    };

    template<class T> struct _Unique_if<T[]> {
        typedef unique_ptr<T[]> _Unknown_bound;
    };

    template<class T, size_t N> struct _Unique_if<T[N]> {
        typedef void _Known_bound;
    };

    template<class T, class... Args>
        typename _Unique_if<T>::_Single_object
        make_unique(Args&&... args) {
            return unique_ptr<T>(new T(std::forward<Args>(args)...));
        }

    template<class T>
        typename _Unique_if<T>::_Unknown_bound
        make_unique(size_t n) {
            typedef typename remove_extent<T>::type U;
            return unique_ptr<T>(new U[n]());
        }

    template<class T, class... Args>
        typename _Unique_if<T>::_Known_bound
        make_unique(Args&&...) = delete;
}

編集: N3656 標準リビジョンにコードを更新

44
sasha.sochka