C++概念成员检查类型与引用的不一致

2022-05-17 00:00:00 c++ c++20 c++-concepts

我正在学习C++概念,我有一个讨厌的问题:

我不知道如何区分成员变量是int类型的变量和成员变量是int&类型。

原因是我正在使用的检查使用的是instance.ember语法,而在C++中,它返回一个引用。

完整示例:

#include <iostream>
#include <concepts>

template<typename T>
void print(T t) {
    std::cout << "generic" << std::endl;
}

template<typename T>
requires requires(T t){
    {t.val} -> std::same_as<int&>;
}
void print(T t) {
    std::cout << "special" << std::endl;
}

struct S1{
    int bla;
};

struct S2{
   int val = 47;
};
int x = 47;
struct S3{
    int& val=x;
};

int main()
{
    print(4.7);
    print(S1{});
    print(S2{});
    print(S3{});
}

我希望print(S3{})由一般情况处理,而不是特殊情况。 请注意,将我的requires内容更改为:

 {t.val} -> std::same_as<int>;

使S2与模板不匹配,因此无法工作(如我所说,我认为C++中的成员访问返回一个引用)。

是否有解决此问题的方法?


解决方案

这里的问题是,表达式概念检查在检查中使用decltype((e)),而不是decltype(e)(额外的圆括号)。

因为t.valint类型的左值(表达式从来没有引用类型),所以decltype((t.val))int&,正如您已经发现的那样。

相反,您需要显式使用Single-Paren语法:

template <typename T>
requires requires (T t) {
    requires std::same_as<decltype(t.val), int&>;
}
void print(T t) {
    std::cout << "special" << std::endl;
}

template <typename T>
requires std::same_as<decltype(T::val), int&>

相关文章