- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在学习模板元编程,最近,我看到了关于 CPPConference 的演讲。关于 void_t。不久之后,我发现了detection idiom .
但是,我仍然很难理解其中任何一个(尤其是检测习惯用法,因为它基于 void_t)。我读过this博客文章,以及this stackoverflow 帖子,这对我有一点帮助,但我仍然有一些问题。
如果我的理解是正确的,如果 void_t 内的表达式无效,则使用以下表达式将其删除:
template< class, class = std::void_t<> >
struct has_type_member : std::false_type { };
因为类在这里是一个默认模板参数,可以代表任意数量的参数,而与它们的类型无关?是否有必要说 class 等于 std::void_t<> ?写得还不够吗
template< class, class = void >
struct has_type_member : std::false_type { };
如果没有,为什么?
但是,如果表达式有效,则该表达式将被调用为 void:
template< class T >
struct has_type_member<T, std::void_t<typename T::type>> : std::true_type { };
为什么有效的表达式会被计算为 void,这对我们有什么帮助?另外,为什么表达式需要有效才能匹配 void_t?
最佳答案
好吧,我并不声称我自己已经完全理解了一切,但我会尽力回答我所知道的:
Because class is here a default template parameter that can represent any number of parameters independent of their type?
差不多了。该模板将匹配具有一两个模板参数的任何实例化,因此所有形式 has_type_member<T>
或has_type_member<T, U>
。这是由于
class
匹配任何类型。但这并不特殊(您也可以写 class T
您只是不需要名称,因为您没有在声明中引用它)每个模板首先匹配所有类型,并且只能通过参数数量来区分。通常,我们要么通过一些 SFINAE 魔法(如 enable_if
)进行约束,要么稍后通过部分模板规范获得更好的拟合。class = void
匹配每种类型,如上所述,并且根本不匹配任何类型,因为 void
如果我们没有争论就填写。我们只会将此模板实例化为 has_member_type<T>
,所以这永远是第一场比赛,但可能不是最好的比赛。但作为第一个匹配,它告诉我们:第二个模板参数必须是 void
,因为所有进一步的匹配必须是部分规范。否则我们就会产生歧义。想一想,如果第二个模板给我们 int
会发生什么?如果表达式有效。然后我们有两场比赛has_type_member<T, void>
和has_type_member<T, int>
,那么我们应该选择哪个呢?这就是为什么在成功案例中类型必须是 void
然后这个重载也被选择,因为它更特殊。
Why would a valid expression be evaluated to void, and how does this help us? Also, why does the expression need to be valid to match void_t?
所以我已经回答了第一个问题的第二部分。关于第一点:想想 void_t
的定义:
template<class...>
using void_t = void;
所以,...
匹配所有内容,无论类型和数量如何,不是吗?实际上它只匹配一个有效的类型,如果不是,它怎么会使用这个类型呢? (我知道它不使用该类型,但它必须能够使用。并且它不能使用无效类型)。因此它给了我们void
传递的模板参数是否有效。所以在我们的用例中:
如果T
具有成员类型 T::type
, T::type
是有效类型且 void_t<...>
匹配它。所以我们得到void_t<T::type>
此时,其值为 void
,适合主要但更特殊,所以我们接受它并得到 true_type
.
如果我们没有类型成员怎么办?然后表达式 T::type
无效,void_t<...>
无法 mazch 它,因此部分规范无效,因此我们无法选择它,但这没问题,因为替换失败不是错误,所以我们继续使用我们已经找到的主模板。
Is it even necessary to say that class equals to std::void_t<> ? Wouldn't it be enough to write
template< class, class = void >
struct has_type_member : std::false_type { };And if not, why?
是的,会的,演讲中也这样做了。 void_t<>
字面意思是void
。我认为void_t
只是为了与第二个规范更加一致。有void_t
是必需的,因为我们需要这个模板。l
关于c++ - SFINAE : Understanding void_t and detect_if,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59276542/
C++11 专家的几个问题。 我正在与 SFINAE 打交道,我遇到了一个奇怪的情况,其中 g++ (4.9.2) 和 clang++ (3.5.0) 的行为不同。 我准备了以下示例代码。很抱歉,我无
这些天我正在试验 SFINAE,有些事情让我很困惑。为什么 my_type_a不能在 my_function 中推导出来的实例化? class my_type_a {}; template clas
我正在尝试复制(我猜)典型的 SFINAE 示例来判断一个类型是否具有特定方法。我的代码基本上是 the one found in the accepted answer of a previous
出于学术原因,我想实现一个示例,如果非类型模板参数满足给定条件,则选择一个模板。例如,我想要一个只为奇数定义的函数。 可以这样做: template struct is_odd: public st
有没有办法检查两个可变参数包的串联是否与第三个可变参数包相同。 template struct ClassOne { } template struct ClassTwo { } template s
为什么以下代码无法编译?为具有特定成员的类型启用模板的最简洁的解决方案是什么?如果模板变量被直接用于初始化它的表达式替换,它也会编译。 #include template constexpr boo
代码: #include using std::nullptr_t; template using nullptr_vt = nullptr_t; struct not_addable{}; tem
我有一个模板类,我想有两个复制 ctor。一个用于平凡类型,另一个用于非平凡类型。 以下代码有效(使用一个拷贝 ctor): template struct MyStruct { MySt
我一直在尝试定义一个辅助类来帮助我使用模板方法,在该方法中我希望为复杂类型和实际类型提供通用实现。 到目前为止,这是我的尝试: #include #include template struct is
这个问题已经有答案了: Officially, what is typename for? [duplicate] (8 个回答) 已关闭 3 年前。 我正在研究现代 C++ 中的 SFINAE,我看
我有一个 std::variants 包含具有不同接口(interface)的对象。目标是如果变体中的对象具有某些方法,则调用它。 我正在尝试制作几个模板装饰器,并寻找一种方法来使用更少的样板和没有宏
我有一个代码,它接受一个函数并根据函数签名执行它,如下所示: template struct Value { int value[Num]; }; struct Executor { t
我一直在尝试定义一个辅助类来帮助我使用模板方法,在该方法中我希望为复杂类型和实际类型提供通用实现。 到目前为止,这是我的尝试: #include #include template struct is
在此视频中https://youtu.be/Vkck4EU2lOU?t=582 “标签调度”和SFINAE作为替代方案出现,允许实现所需模板功能的选择。 正确吗? “标签发送”不是使用SFINAE吗?
我认为下面的代码会编译,因为冲突的重载是 SFINAEd 了。但是编译器(GCC)说:void Foo::bar(Xd) const' cannot be overloaded .有没有简单的方法来修
#define BINDINGTEMPLATE template, int> || std::is_same_v, std::string> || std::is_same_v, char>>> 这样
SFINAE 是否在概念论证中起作用? (也许这里不叫 SFINAE)。例子: template requires std::invocable && // , void>)
在他的演讲现代模板元编程:纲要第 I 部分中,Walter Brown 以他的方式讨论了 enable_if 与 SFINAE 的交互。 在大约 47:40 的谈话中,他被问到一个问题,我无法完全匹配
我正在尝试对同一函数进行两次重载,称为某物。这个函数应该以另一个函数作为参数,并且它应该根据这个另一个函数的返回类型进行重载。到目前为止我有这个: #include #include using
以下代码特化了 f() 的两个版本。第一个检测一个 vector 并返回一个迭代器。第二个接受所有其他类型并返回一个拷贝。 这无法在 VC 2010 上编译,GetIter2 中有一个错误,即 Get
我是一名优秀的程序员,十分优秀!