- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我已经使用 CRTP 为插件系统构建了一个自动注册框架,效果很好。当我想拥有它的继承历史时,问题就来了。我最初的计划是在类自动注册时累积父 ID 的 vector 。我的尝试在下面的代码中,因为它相当长,所以我把它放在问题的末尾。
InterfaceFactory 和 AutoRegister 类通过 CRTP 工作正常(即使 AutoRegister 功能被多次继承)。系统能够根据提供的 id 创建正确的类型,并且所有注册功能都是由编译器在编译时构建的。
我想添加继承历史是在他们注册时添加继承,看起来很简单。我计划在注册期间使用静态“InterfaceType s_type”来获取 INTERFACE 类的信息并添加到当前。问题是原始接口(interface)调用“BaseInterface”(和类似的接口(interface))没有 s_type。接下来的想法是使用 SFINAE 忽略层次结构中的那些类(因为它无论如何都没有 id)。这导致了 AutoRegister 中的 BaseId 结构,它试图关闭 s_type。然而,这似乎不起作用,因为从未使用过特化。我假设 Base::BaseId 是使用非专用的创建的,因为没有 BaseInterface::s_type 并且当 Derived 被注册时(因为它是从 Base 派生的)编译器正在获取 Base::BaseId 并且甚至没有尝试特化。所以我的问题是,这个假设是否正确?如果可以,有人可以稍微解释一下吗,因为我对细节不太清楚,还有其他方法可以做到这一点吗?
#include <vector>
#include <functional>
template <typename T> struct TypeName
{
static std::string get()
{
std::string fullName=typeid(T).name();
size_t beginPos=0;
size_t classPos=fullName.find("class");
size_t nameSpacePos=fullName.find_last_of("::");
if(classPos != std::string::npos)
beginPos=classPos+6;
if(nameSpacePos != std::string::npos)
{
if(nameSpacePos+1 > beginPos)
beginPos=nameSpacePos+1;
}
return fullName.substr(beginPos);
}
};
class BaseInterface
{
public:
BaseInterface() {}
virtual ~BaseInterface() {}
};
struct InterfaceType
{
InterfaceType():id(-1){}
unsigned int id;
std::vector<unsigned int> idInheritance;
std::string name;
};
template<typename CLASS, typename INTERFACE>
class AutoRegister:public INTERFACE
{
public:
AutoRegister():INTERFACE() { &s_type; }
static BaseInterface *create(unsigned int type) { CLASS *newClass=new CLASS(); return newClass; }
template<typename T, typename=int>
struct BaseId
{
static std::vector<unsigned int> get() { return std::vector<unsigned int>(); }
};
template<typename T>
struct BaseId<T, decltype((void)T::s_type, 0)>
{
static std::vector<unsigned int> get()
{
std::vector<unsigned int> ids;
ids.push_back(T::s_type.id); ids.insert(ids.end(), T::s_type.idInheritance.begin(), T::s_type.idInheritance.end());
return ids;
}
};
static std::vector<unsigned int> getBaseIds() { return BaseId<INTERFACE>::get(); }
private:
static InterfaceType s_type;
};
class InterfaceFactory
{
public:
typedef BaseInterface *(*FactoryFunc)(unsigned int type);
class BaseInterfaceNode
{
public:
BaseInterfaceNode(unsigned int type, std::string typeName, FactoryFunc factoryFunction):
m_type(type), m_typeName(typeName), m_factoryFunction(factoryFunction)
{};
size_t type() { return m_type; }
std::string typeName() { return m_typeName; }
BaseInterface *factoryFunction() { return m_factoryFunction(m_type); }
private:
unsigned int m_type;
std::string m_typeName;
FactoryFunc m_factoryFunction;
};
typedef std::vector<BaseInterfaceNode> BaseInterfaceNodes;
public:
InterfaceFactory() {}
~InterfaceFactory() {}
public:
static BaseInterface *createType(unsigned int type);
static InterfaceType registerType(std::string typeName, std::vector<unsigned int> ids, FactoryFunc factoryFunc);
struct BaseInterfaceNodeHolder
{
BaseInterfaceNodeHolder():s_interfaceTypeIndex(0) {}
BaseInterfaceNodes s_baseInterfaceNodes;
unsigned int s_interfaceTypeIndex;
};
private:
};
template<typename CLASS, typename INTERFACE> InterfaceType AutoRegister<CLASS, INTERFACE>::s_type=\
InterfaceFactory::registerType(TypeName<CLASS>::get(), AutoRegister<CLASS, INTERFACE>::getBaseIds(), &AutoRegister<CLASS, INTERFACE>::create);
InterfaceFactory::BaseInterfaceNodeHolder *s_nodes=nullptr;
InterfaceType InterfaceFactory::registerType(std::string typeName, std::vector<unsigned int> ids, FactoryFunc factoryFunc)
{
if(s_nodes == nullptr)
s_nodes=new BaseInterfaceNodeHolder();
InterfaceType sampleType;
sampleType.id=s_nodes->s_interfaceTypeIndex;
sampleType.idInheritance=ids;
sampleType.name=typeName;
s_nodes->s_baseInterfaceNodes.push_back(BaseInterfaceNode(s_nodes->s_interfaceTypeIndex, typeName, factoryFunc));
s_nodes->s_interfaceTypeIndex++;
return sampleType;
}
//////////////////////////////////////////////////////////////////////
class Base:public AutoRegister<Base, BaseInterface>
{
};
class Derived:public AutoRegister<Derived, Base>
{
};
int main(int argc, const char* argv[])
{
Base base;
Derived derived;
return 0;
}
谢谢,
最佳答案
好吧,我发现了我自己的错误,似乎 SFINAE 的风格要求我检查的成员变量是公开的,一旦我改变了
private:
static InterfaceType s_type;
到
public:
static InterfaceType s_type;
它解决了问题,至少对于 Clang(可能还有 gcc)来说是这样。但是 VS2013 没有工作。阅读了一下之后,当涉及到一些高级模板时,VS2013 中似乎可能存在一些错误。大多数似乎已修复但尚未发布。最后我不得不使用 Johannes Schaub - litb ( How to detect whether there is a specific member variable in class? ) 提供的版本。我包含了一个 has_s_type
template <typename T>
struct has_s_type
{
struct Fallback { InterfaceType s_type; };
struct Derived: T, Fallback {};
template<typename C, C> struct ChT;
template<typename C> static char(&f(ChT<InterfaceType Fallback::*, &C::s_type>*))[1];
template<typename C> static char(&f(...))[2];
static bool const value=sizeof(f<Derived>(0)) == 2;
};
并将AutoRegister修改为
template<typename CLASS, typename INTERFACE>
class AutoRegister:public INTERFACE
{
public:
AutoRegister():INTERFACE() { &s_type; }
static BaseInterface *create(unsigned int type) { CLASS *newClass=new CLASS(); return newClass; }
template<typename T, bool=false>
struct BaseId
{
static std::vector<unsigned int> get() { return std::vector<unsigned int>(); }
};
template<typename T>
struct BaseId<T, true>
{
static std::vector<unsigned int> get()
{
std::vector<unsigned int> ids;
ids.push_back(INTERFACE::s_type.id); ids.insert(ids.end(), INTERFACE::s_type.idInheritance.begin(), INTERFACE::s_type.idInheritance.end());
return ids;
}
};
static std::vector<unsigned int> getBaseIds() { return BaseId<INTERFACE, has_s_type<INTERFACE>::value>::get(); }
public:
static const InterfaceType s_type;
};
这是一些live code
关于c++ - 具有继承历史的 CRTP 插件自动注册(尝试但失败了 SFINAE),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30518410/
我想做的是,如果鼠标位于“下一个”按钮上,它会以慢速向右滚动,如果鼠标没有位于“下一个”按钮上,它会停止滚动? 这是我的尝试http://jsfiddle.net/mdanz/nCCRy/14/ $(
StyleCop 是一个很棒的视觉工作室小插件。但它不会向您显示实时提示或提供任何自动修复。 随之而来的是 reSharper 和 StyleCop for reSharper,这是理想的解决方案,但
我为我的MatchQuery使用了模糊性选项,但是我想将模糊性值设置为auto。有什么办法吗? 另外,对于完成建议程序,您可以将其设置为支持unicode,对于我的MatchQuery,有什么方法可以
我想从表中获取一行[字符串名称,字符串密码,int 某些内容]并将其映射到一个 User 对象,该对象具有 3 个属性,如上面的 getter 和 setter有什么方法可以自动完成吗?我考虑过反射,
我有一个像这样的方法:void m1(string str) 并且有一个像这样的类: public class MyClass { public bool b1 { set; get; }
我正在尝试使用 $rootScope 从一个 Controller 向另一个 Controller $broadcast 一些数据。 如果我使用像 ng-click 这样的触发器来运行将广播的功能,它
我考虑了很多关于是要使用完全自动化的缓存还是手动缓存。 我们的自动方法是一种解决方案,它可以挖掘数据库、查询和格式化每个潜在和 future 的数据请求,并将其保存到适当的缓存存储(内存缓存或基于磁盘
我的 CSS 必须使用过渡来更改,直到现在我都使用 div:hover 来实现。 当您单击另一个 div 时需要激活过渡,而不是当您将鼠标悬停在必须移动/更改的 div 上时。 我该怎么做? 谢谢 永
在我的应用程序中,我需要一些动画,但如果它已经设置了动画,则不需要持续时间。但我的问题是它会自动添加持续时间。 在这里你可以看到 2 个函数,第二个没有持续时间但它确实有持续时间(可能从 1 秒开始)
两年前,我需要制作一个工具,通过 POST 自动将 txt/csv 文件上传到我的 Web 服务器,然后使用 cronjob 通过 PHP 对其进行解析。 这有两次在每天午夜自动发生。尽管这行得通,但
请阅读下面程序中的评论: #include void test(char c[]) { c=c+2; //why does this work ? c--; printf("%
也许是个幼稚的问题,但是...... 确认或拒绝: 自动和静态存储持续时间的对象/变量的内存的存在是在编译时确定的,程序运行时失败的可能性绝对为零,因为没有足够的内存用于自动对象。 自然地,当自动对象
有没有什么方法可以自动获得类中属性更改的通知,而不必在每个 setter 中都编写 OnPropertyChanged? (我有数百个属性,我想知道它们是否已更改)。 安东建议 dynamic pro
我们在使用 Azure DevOps 的项目中采用了 gitflow 流程。我有以下场景: 当功能分支合并到 Develop 时,我想在完成拉取请求的同时执行压缩合并策略 当 Release 分支定期
我的网站上有一个评论部分,我将 html 编码的评论保存在我的数据库中。所以我添加了这条评论- "testing" `quotes` \and backslashes\ and html 并将其保存在
是否存在“ checkin 前 TFS 自动 checkout ”这样的功能,以便在我说“ checkin ”之前我不会 checkout 任何文件,例如以防我只是临时更改文件 - 这一直发生。 换句
我有一个运行在 Linux/Apache/Tomcat 堆栈上的网站,它需要每隔几个月自动脱机以进行服务器维护,这将持续任意时间。有哪些选项可以让 Apache 建立和取消“服务器维护”页面? 我需要
我经常在工作中创建文档,在公司内部,由于我们使用的首字母缩写词和缩写词的数量,我们几乎拥有自己的语言。因此,我厌倦了在发布文档之前手动创建首字母缩写词和缩写表,并且快速的谷歌搜索发现了一个可以有效地为
我希望在用户或宏将计算模式从自动更改为手动或手动更改为自动时运行代码。是否有为此触发的事件? (属性是 Application.Calculation 在 Excel 互操作中。) 使用 Excel
这个问题在这里已经有了答案: Repeat command automatically in Linux (13 个回答) 6年前关闭。 我想创建一个脚本来获取另一个文件夹中的所有文件夹名称。并为这些
我是一名优秀的程序员,十分优秀!