Разное поведение моего type_trait в шаблонном / не шаблонном коде - PullRequest
4 голосов
/ 11 июня 2019

В следующем фрагменте has_bar ведет себя по-разному в основном и в методе DoStuff:

В основном методе a_bar == false и b_bar == true.

Когда я выполняю это, я получаю 2x "Foo" в качестве вывода. Почему?

#include <iostream>

struct A
{
    void Foo() { std::cout << "Foo" << std::endl; }
};

struct B : public A
{
    void Bar() {  std::cout << "Bar" << std::endl; }
};

template<typename, typename = void>
struct has_bar : std::false_type
{ };

template<typename T>
struct has_bar<T, std::void_t<decltype(T::Bar)>> : std::true_type
{ };

template<typename T>
void DoStuff(T t)
{
    if constexpr (has_bar<T>::value)
    {
        t.Bar();
    }
    else
    {
        t.Foo();
    }
}


int main()
{
    A a;
    B b;

    constexpr bool a_bar = has_bar<A>::value; // false
    constexpr bool b_bar = has_bar<B>::value; // true

    DoStuff(a);
    DoStuff(b);

    std::cin.ignore();

    return 0;
}

1 Ответ

7 голосов
/ 11 июня 2019

Должно быть:

template<typename T>
struct has_bar<T, std::void_t<decltype(&T::Bar)>> : std::true_type
//                                     ^^
{ };

Демо

...