2011-03-15 216 views
10

我該如何懶惰評估std :: conditional中的第二個參數?懶惰評價

#include "stdafx.h" 
#include <type_traits> 

struct Null{}; 
struct _1{enum {one = true,two = false};}; 
struct _2{enum {two = true, one = false};}; 

template<class T> 
struct is_nulltype 
{ 
    enum {value = false}; 
}; 

template<> 
struct is_nulltype<Null> 
{ 
    enum {value = true}; 
}; 

template<class T> 
struct X : std::conditional<is_nulltype<T>::value,Null,typename std::conditional<T::one,_1,_2>::type>::type 
{ 
}; 

int _tmain(int argc, _TCHAR* argv[]) 
{ 
X<Null> x;//won't compile no Null::one but I don't need that member in Null at all 
    return 0; 
} 
+1

對於手頭的問題,只需將'X'專用於'Null'類型。 – 2011-03-15 19:38:10

+0

@Alf這意味着我必須有兩個「幾乎」相同的類。不行。 – 2011-03-15 19:47:46

+0

它看起來像你在這裏尋找的是「懶惰編譯」,而不是「懶惰評估」。在X的定義中將T替換爲T會導致語法錯誤。語法的評估不能被延遲。 – goedson 2011-03-15 20:16:41

回答

9

這種情況的常用的方法是有std::conditional 2元功能之間進行選擇:兩個::type小號std::conditional後現在

template <typename T> 
struct false_case { 
    typedef typename std::conditional<T::one,_1,_2>::type type; 
}; 

struct always_null {typedef Null type;}; 

template<class T> 
struct X : 
    std::conditional<is_nulltype<T>::value, 
        always_null, 
        false_case<T> 
        >::type::type { ... }; 

注意。

+0

'false_case'可以將'conditional'作爲基類來縮短它的長度。 – aaz 2011-03-15 20:33:24

+0

謝謝,我認爲它會做我想做的。如果沒有,我會讓你知道。 – 2011-03-15 20:45:36