名前空間
変種
操作

std::unwrap_reference, std::unwrap_ref_decay

From cppreference.com
< cpp‎ | utility‎ | functional
 
 
ユーティリティライブラリ
言語サポート
型のサポート (基本型、RTTI)
ライブラリ機能検査マクロ (C++20)
プログラムユーティリティ
可変引数関数
コルーチンサポート (C++20)
契約サポート (C++26)
三方比較
(C++20)
(C++20)(C++20)(C++20)  
(C++20)(C++20)(C++20)

汎用ユーティリティ
関係演算子 (C++20で非推奨)
 
関数オブジェクト
関数の呼び出し
(C++17)(C++23)
恒等関数オブジェクト
(C++20)
参照ラッパー
(C++11)(C++11)
unwrap_referenceunwrap_ref_decay
(C++20)(C++20)
透過的な演算子ラッパー
(C++14)
(C++14)
(C++14)
(C++14)  
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)

古いバインダとアダプタ
(C++17まで*)
(C++17まで*)
(C++17まで*)
(C++17まで*)  
(C++17まで*)
(C++17*まで)(C++17*まで)(C++17*まで)(C++17*まで)
(C++20まで*)
(C++20まで*)
(C++17*まで)(C++17*まで)
(C++17*まで)(C++17*まで)

(C++17まで*)
(C++17*まで)(C++17*まで)(C++17*まで)(C++17*まで)
(C++20まで*)
(C++20まで*)
 
ヘッダ <type_traits> で定義
ヘッダ <functional> で定義
template< class T >
struct unwrap_reference;
(1) (C++20以降)
template< class T >
struct unwrap_ref_decay;
(2) (C++20以降)

あらゆる std::reference_wrapper をアンラップします: std::reference_wrapper<U>U& に変更します。

1) Tstd::reference_wrapper の特殊化である場合、それをアンラップします。それ以外の場合、T は変化しません。
2) 崩壊した Tstd::reference_wrapper の特殊化である場合、それをアンラップします。それ以外の場合、T は崩壊します。

プログラムがこのページで説明されているテンプレートのいずれかに特殊化を追加する場合、動作は未定義です。

目次

[編集] 入れ子型

定義
type

(1) Tstd::reference_wrapper<U> の場合、U&。それ以外の場合、T
(2) std::decay_t<T>std::reference_wrapper<U> の場合、U&。それ以外の場合、std::decay_t<T>

[編集] ヘルパー型

template<class T>
using unwrap_reference_t = unwrap_reference<T>::type;
(1) (C++20以降)
template<class T>
using unwrap_ref_decay_t = unwrap_ref_decay<T>::type;
(2) (C++20以降)

[編集] 実装例

template<class T>
struct unwrap_reference { using type = T; };
template<class U>
struct unwrap_reference<std::reference_wrapper<U>> { using type = U&; };
 
template<class T>
struct unwrap_ref_decay : std::unwrap_reference<std::decay_t<T>> {};

[編集] 備考

std::unwrap_ref_decay は、std::make_pair および std::make_tuple で使用される変換と同じ変換を実行します。

機能テストマクロ 規格 機能
__cpp_lib_unwrap_ref 201811L (C++20) std::unwrap_ref_decaystd::unwrap_reference

[編集]

#include <cassert>
#include <functional>
#include <iostream>
#include <type_traits>
 
int main()
{
    static_assert(std::is_same_v<std::unwrap_reference_t<int>, int>);
    static_assert(std::is_same_v<std::unwrap_reference_t<const int>, const int>);
    static_assert(std::is_same_v<std::unwrap_reference_t<int&>, int&>);
    static_assert(std::is_same_v<std::unwrap_reference_t<int&&>, int&&>);
    static_assert(std::is_same_v<std::unwrap_reference_t<int*>, int*>);
 
    {
        using T = std::reference_wrapper<int>;
        using X = std::unwrap_reference_t<T>;
        static_assert(std::is_same_v<X, int&>);
    }
    {
        using T = std::reference_wrapper<int&>;
        using X = std::unwrap_reference_t<T>;
        static_assert(std::is_same_v<X, int&>);
    }
 
    static_assert(std::is_same_v<std::unwrap_ref_decay_t<int>, int>);
    static_assert(std::is_same_v<std::unwrap_ref_decay_t<const int>, int>);
    static_assert(std::is_same_v<std::unwrap_ref_decay_t<const int&>, int>);
 
    {
        using T = std::reference_wrapper<int&&>;
        using X = std::unwrap_ref_decay_t<T>;
        static_assert(std::is_same_v<X, int&>);
    }
 
    {
        auto reset = []<typename T>(T&& z)
        {
        //  x = 0; // Error: does not work if T is reference_wrapper<>
            // converts T&& into T& for ordinary types
            // converts T&& into U& for reference_wrapper<U>
            decltype(auto) r = std::unwrap_reference_t<T>(z);
            std::cout << "r: " << r << '\n';
            r = 0; // OK, r has reference type
        };
 
        int x = 1;
        reset(x);
        assert(x == 0);
 
        int y = 2;
        reset(std::ref(y));
        assert(y == 0);
    }
}

出力

r: 1
r: 2

[編集] 関連項目

コピー構築可能 (CopyConstructible) かつ コピー代入可能 (CopyAssignable) な参照ラッパー
(クラステンプレート) [編集]
引数の型によって決定される型のpairオブジェクトを生成する
(関数テンプレート) [編集]
引数型によって定義された型の tuple オブジェクトを生成する
(関数テンプレート) [編集]
English 日本語 中文(简体) 中文(繁體)