- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我的目标是实现一个检测嵌套 using
是否存在的谓词别名(或 typedef
)充当轻量级标签以指示类具有某些属性(用于泛型编程)。例如,has_my_tag<T>
谓词的行为应如下所示:
struct A {
using my_tag = void;
};
struct B {};
int main()
{
static_assert(has_my_tag<A>::value, ""); // evaluate to true if my_tag=void is present
static_assert(!has_my_tag<B>::value, ""); // false otherwise
}
用户@JoelFalcou 称其为“轻量级类型分类成语”并在 this answer 中提供了解决方案.我找不到任何关于那个名字的成语的引用资料(你知道吗?)这是乔尔对 has_my_tag<>
的实现。 :
template<class T, class R = void>
struct enable_if_type { typedef R type; };
template<class T, class Enable = void>
struct has_my_tag : std::false_type {};
template<class T>
struct has_my_tag<T, typename enable_if_type<typename T::my_tag>::type> :
std::true_type
{};
这里是编译器资源管理器上的工作版本:https://godbolt.org/z/EEOBb-
我想出了以下简化版本:
template<class T, class Enable = void>
struct has_my_tag : std::false_type {};
template<class T>
struct has_my_tag<T, typename T::my_tag> : std::true_type
{};
我的问题:简化版本是一种可以接受的实现习语的方式吗?在某些情况下它会失败吗?是否有适用于 C++11 的更简单版本?我应该更喜欢哪个版本?
据我了解,Joel 的版本将允许 my_tag
为任何类型起别名,而我的版本需要 my_tag
别名 void
.但考虑到为轻量级谓词测试标记类型的目标,我不清楚哪个版本是首选。
辅助问题:还有,这个成语还有别的名字吗?它是否在我可以调查的任何图书馆中使用?到目前为止,我还没有找到可以显示任何搜索结果的名称。
最佳答案
对于您的设置,原始版本和您的没有区别。两者都使用 SFINAE 选择正确的 has_my_tag。但是,您的版本确实限制了您的 typedef/using
成为my_tag=void
.如果 my_tag 与任何其他类型一样是 typedef,则您的特化将不匹配,并且您最终将实例化主模板,如图所示 here .
这样做的原因是您在 main 中实例化模板的位置,static_assert(has_my_tag<A>::value, "");
您没有指定第二个参数,因此使用默认值 (void),即 has_my_tag<A,void>::value
您的专业必须与此相匹配才能被考虑。
enable_if_type
的用法,(基本上是在 c++17 中完成 void_t 的工作)是在 ::type
上启用 SFINAE T 的成员,但总是导致无效,这样您的特化将在 ::type
时匹配存在,无论 my_tag
的类型如何类型定义。
这让您只需担心它是否存在,而不是它的类型;
我个人会使用不依赖于 my_type
的方法被 typedef 定义为 void,要么是 enable_if_type 版本,要么是类似...
#include <iostream>
#include <type_traits>
struct A {
using my_tag = void;
};
struct B {};
struct C {
using my_tag = int; // with void_t also works with my_tag = int
};
struct D {
struct my_tag{}; //or some struct as the tag
};
// same as your enable_if_type
template <typename...>
using void_t = void;
template<class T, class Enable = void>
struct has_my_tag : std::false_type {};
template<class T>
struct has_my_tag<T, void_t<typename T::my_tag>> : std::true_type
{};
int main() {
std::cout << has_my_tag<A>::value << std::endl;
std::cout << has_my_tag<B>::value << std::endl;
std::cout << has_my_tag<C>::value << std::endl;
std::cout << has_my_tag<D>::value << std::endl;
return 0;
}
关于c++ - "lightweight type categorization idiom"的最简单实现?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52777854/
有人指出我是“安全 bool 惯用语”,在尝试破译发生了什么之后(解释 supplied on the site 不足以让我理解为什么它有效),我决定尝试将以下代码分开并尝试尽可能地简化它。该站点提供
我想使用 Pimpl Idiom,但我遇到了一个问题,其中一个成员函数是模板函数,因此它必须在头文件中实现。 例如下面这个当然可以正常工作 //Foo.h class Foo{ struct
假设我有一个 C 库,它有自己的(反)初始化例程,很多库都有。 init_API(); deinit_API(); 现在假设我想为用户提供另一个抽象级别,并使用静态实例化的类抽象掉这些调用。我想到的方
我知道在 C 中看起来像这样的 pimpl-idiom: // foobar.h struct FooBar { char *someString; struct FooBarImpl
我有以下代码的错误 incomplete type ‘Foo::Pimpl’ used in nested name specifier 另一个Foo.hpp struct AnotherFoo {
考虑以下 try-with-resources block : try (Foo foo = getAFoo()) { } 对于某些实现了 java.lang.AutoCloseable 的类 Foo
我们有一个客户希望访问的高度模板化的仅 header 代码库。例如,假设它包含 Foo标题中的类 foo.hpp : #ifndef FOO_HEADER #define FOO_HEADER #in
D 有一个很棒的模块系统,与 C++ 相比,它大大减少了编译时间。根据文档 D 仍然提供不透明的结构和联合以启用 pimpl 习惯用法。我的问题是:如何在一个模块中声明嵌套结构(或联合)并在另一个模块
我基本上有这段代码: for (var i = 0; i < num; i++) { ShowCard(i); } 现在,ShowCard 只是添加一个 DOM 元素,但我希望它有一个动画,显示这
我正在尝试为将使用 pimpl-idiom 的库定义接口(interface)。下面是我定义的一个典型的接口(interface)类。 struct A { public: void func1
注意:此问题遵循 a previous one , 我希望仍然可以将其作为一个新问题提出。 我正在尝试为树类实现“三个半大规则”( copy-and-swap 习语),它看起来像这样: class T
好的。所以我有一个值列表,我想执行如下操作: MyObjectValues .Select(currentItems=>new MyType() { Parameter1 = current
通常的克隆习语使用协变返回类型: struct Base { virtual Base* clone(); }; struct Derived : public Base { Deri
传统的PImpl Idiom是这样的: #include struct Blah { //public interface declarations private: struct
我正在阅读一本关于 Swift 设计模式的书,并遇到了一个示例,作者希望确保任何初始化特定类的子类的人都必须传入基类所需的值: class Employee { private var pro
我刚被一个 bug 搞得焦头烂额,部分原因是我缺乏理解,部分原因是我认为我们的代码库中的设计欠佳。我很好奇如何改进我的 5 分钟解决方案。 我们正在使用引用计数对象,我们在这些类的对象上使用 AddR
这个问题在这里已经有了答案: Use of void template argument in early detection idiom implementation (1 个回答) 关闭 5 年
正如标题所暗示的,我想知道除了减少重建时间之外,PImpl 惯用语还有哪些其他原因、目的或用途。 引用 here 中的示例: (thread locals need to be behind a pi
PIMPL 习语的目的是隐藏实现,包括方法、结构,甚至结构的大小。一个缺点是它使用堆。 但是,如果我不想隐藏任何东西的尺寸要求怎么办。我只是想隐藏方法、结构的格式和变量名。一种方法是分配一个完美大小的
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我是一名优秀的程序员,十分优秀!