通过类模板参数

时间:2016-10-10 05:24:59

标签: c++ templates gcc visual-studio-2015 clang

我试图编写一个模板化的类,它本质上只是其他类型的包装器,以便能够轻松地观察值发生的事情 - 每次任何操作符,构造函数,析构函数等, 叫做。 所以我试图创建一个运算符bool(),它通过在任何没有运算符bool()的类上有一个static_cast的类调用运算符bool()来工作。如果有人试图转换为bool,那么无法执行任何操作的类应该只会失败编译。我还没有专注于这一部分,因为我对这个更简单的案例感到难过。

它在MSVC 2015上编译并正常工作,但在Clang(3.8.1)或GCC(6.2)上都没有编译。 GCC和Clang都设置为-std=c++1z

链接到在线编译器,显示GCC和Clang上的错误:https://godbolt.org/g/v3B6TE

这是一个显示问题的精简版本:

#include <type_traits>
#include <utility>
#include <assert.h>

template <typename T>
struct Wrapper {
    T val;

    template<typename...Params>
    Wrapper(Params&&...args): val(std::forward<Params>(args)...){}

    /**
    * \brief bool conversion operator
    */
    template <typename = std::enable_if_t<std::is_fundamental<T>::value>>
    operator bool() const {
        return static_cast<bool>(val);
    }

    /**
    * \brief bool conversion operator
    */
    template <typename = std::enable_if_t<!std::is_fundamental<T>::value>, typename = void>
    operator bool() const {
        return val.operator bool();
    }
};


struct HasOperatorBool {
    mutable bool done{ false };

    operator bool() const{
        done = true;
        return done;
    }
};

int main(int argc, char** argv) {
    Wrapper<HasOperatorBool> whob;
    bool didIt = whob;
    assert(didIt);

    Wrapper<int> wi{1};
    bool bi = wi;
    assert(bi);

    return 0;
}

clang和gcc都在说std::enable_if_t没有type

GCC

In file included from /tmp/gcc-explorer-compiler116910-70-16kehep/example.cpp:1:
/usr/lib/gcc/x86_64-linux-gnu/5.4.1/../../../../include/c++/5.4.1/type_traits:2388:44: error: no type named 'type' in 'std::enable_if<false, void>'; 'enable_if' cannot be used to disable this declaration
using enable_if_t = typename enable_if<_Cond, _Tp>::type;
^~~~~
15 : note: in instantiation of template type alias 'enable_if_t' requested here
template <typename = std::enable_if_t<std::is_fundamental<T>::value>>
^
40 : note: in instantiation of template class 'Wrapper<HasOperatorBool>' requested here
Wrapper<HasOperatorBool> whob;
^
In file included from /tmp/gcc-explorer-compiler116910-70-16kehep/example.cpp:1:
/usr/lib/gcc/x86_64-linux-gnu/5.4.1/../../../../include/c++/5.4.1/type_traits:2388:44: error: no type named 'type' in 'std::enable_if<false, void>'; 'enable_if' cannot be used to disable this declaration
using enable_if_t = typename enable_if<_Cond, _Tp>::type;
^~~~~
2 errors generated.
Compiler exited with result code 1

铛:

In file included from /tmp/gcc-explorer-compiler116910-70-1nlkefa/example.cpp:1:0:
/opt/gcc-explorer/gcc-6.2.0/include/c++/6.2.0/type_traits: In substitution of 'template<bool _Cond, class _Tp> using enable_if_t = typename std::enable_if::type [with bool _Cond = std::is_fundamental<HasOperatorBool>::value; _Tp = void]':
16 : required from 'struct Wrapper<HasOperatorBool>'
40 : required from here
/opt/gcc-explorer/gcc-6.2.0/include/c++/6.2.0/type_traits:2512:61: error: no type named 'type' in 'struct std::enable_if<false, void>'
using enable_if_t = typename enable_if<_Cond, _Tp>::type;

我可能对替代方法感兴趣,但我对技术原因非常感兴趣,因为它不能与3个编译器中的2个一起工作。一个或多个编译器是错误的吗?有没有办法达到我想要的目的?

1 个答案:

答案 0 :(得分:3)

你实际上并没有做SFINAE。使用std::enable_if<T>时,有问题的T应直接来自编译器尝试实例化的模板( immediate-context )。但由于T的模板参数,而不是方法,因此替换失败并提供硬错误而不是SFINAE。我不确定VS的行为在这方面是否符合标准。

您想要的是引入一个默认为类模板参数的虚拟模板参数:

/**
* \brief bool conversion operator
*/
template <typename X = T, typename = std::enable_if_t<std::is_fundamental<X>::value>>
operator bool() const {
    return static_cast<bool>(val);
}

/**
* \brief bool conversion operator
*/
template <typename X = T, typename = std::enable_if_t<!std::is_fundamental<X>::value>, typename = void>
operator bool() const {
    return val.operator bool();
}