我想将uuid / guid与模板专业化相关联。
以下代码可用于将uuid与非模板接口(class,struct)相关联:
__interface __declspec(uuid("CECA446F-2BE6-4AAC-A117-E395F27DF1F8")) ITest {
virtual void Test() = 0;
};
GUID guid = __uuidof(ITest); // OK
现在我有一个模板化界面
template<class T> __interface ITemplateTest {
virtual void Test(T t) = 0;
};
我想做以下工作:
GUID templateGuid = __uuidof(ITemplateTest<float>);
无法将__declspec(uuid(...))添加到模板类定义中。这是显而易见的,因为界面的不同特化需要不同的uuids。所以我尝试通过以下方式将uuid与模板特化相关联:
template<> __interface __declspec(uuid("CF4AB938-8CE0-4AB7-A56C-0253B6018C26")) ITemplateTest<float>;
不幸的是,这也不起作用(__uuidof(。)失败,“没有GUID与此对象相关联”。)
我的问题有解决办法吗?
以下是一些更完整的示例,稍后会在一些答案之后添加:
假设我有一个适用于不同数据类型的复杂算法。为简单起见,这种“复杂”算法正在对数字进行平方。我只想实现我的算法一次,所以模板是要走的路。让我们进一步假设我想使用接口,因为我的应用程序使用COM。
所以这是一个接口和一个实现这个接口的模板化对象:
template<class T> __interface ITemplateTest {
virtual T Square(T t) = 0;
};
template<class T> class CTemplateImplementation : public ITemplateTest<T> {
public:
virtual T Square(T t) { return t * t; };
};
这允许做类似
的事情CTemplateImplementation<double> xDouble;
CTemplateImplementation<float> xFloat;
CTemplateImplementation<int> xInt;
std::cout << xDouble.Square(5.) << std::endl
<< xFloat.Square(5.0f) << std::endl
<< xInt.Square(5) << std::endl;
现在让我们进一步假设我有另一个模板对象,也实现了一个非常“复杂”的算法,它利用了CTemplateImplementation:
template<class T> class CSquareAndAddOne {
private:
CTemplateImplementation<T> m_squarer;
public:
T SquareAndAddOne(T t) { return m_squarer.Square(t) + T(1); }
};
现在可以以相同的方式使用此对象:
CSquareAndAddOne<double> yDouble;
CSquareAndAddOne<float> yFloat;
CSquareAndAddOne<int> yInt;
std::cout << yDouble.SquareAndAddOne(5.) << std::endl
<< yFloat.SquareAndAddOne(5.0f) << std::endl
<< yInt.SquareAndAddOne(5) << std::endl;
当CSquareAndAddOne :: SquandAndAddOne想要使用CTemplateImplementation专门化的__uuid时,问题就出现了。请尝试以下方法:
T SquareAndAddOne(T t) {
GUID guid = __uuidof(m_multiplier);
return m_squarer.Square(t) + T(1);
}
这不再起作用,因为没有与m_multiplier相关的GUID。那么如何在没有重复代码的情况下为这些(在这种情况下为三个)CTemplateImplementation的不同实现分配guid?你能提供完整的解决方案吗? 从CTemplateImplementation为不同类型派生不同的类是不可能的,因为CSquareAndAddOne中正确类型化的专业化的使用不能再用模板参数控制。
答案 0 :(得分:3)
我尝试了一些明确的实例化魔法,它运行正常(至少在VS2008 SP1中):
template<class T> __interface ITemplateTest { void Test(T t); }; // Interface declaration
template __interface ITemplateTest<float>; // Explicit instantiation
template __interface __declspec(uuid("CF4AB938-8CE0-4AB7-A56C-0253B6018C26") ITemplateTest<float>; // Repeat explicit instantiation with uuid association
GUID guid = __uuidof(ITemplateTest<float>); // Enjoy! :-)
看起来最初的问题是__declspec
尝试在实际实例化之前将guid分配给类专门化。
答案 1 :(得分:0)
尝试:
__interface __declspec(uuid("CF4AB938-8CE0-4AB7-A56C-0253B6018C26"))
ITestFloat: ITemplateTest<float> {
virtual void Test() = 0;
};