将模板专门化为GUID值

Specialize a template for a GUID value

本文关键字:GUID 专门化      更新时间:2023-10-16

这是为什么常量POD对象的字段本身不是常量?

库的头文件声明了像

这样的类guid
static const GUID CLSID_EH264VD = 
{ 0x96b9d0ed, 0x8d13, 0x4171, { 0xa9, 0x83, 0xb8, 0x4d, 0x88, 0xd6, 0x27, 0xbe } };

我想写一个直接从dll创建对象的函数,不需要注册dll,所以我需要将每个CLSID映射到dll名称。就像

Create<CLSID_EH264VD>()

依赖于像

这样的专门化
template<>
struct dll<CLSID_EH264VD>
{
    char const* filename = ""mc_dec_avc_ds.ax";
}

使它是一个编译时错误,试图实例化一个未注册的类与一个未知的dll。

问题是模板不能专门用于guid。相关的问题说constexpr将允许以一种允许专门化的方式声明GUID,但Visual c++在最新版本(2012)中不支持constexpr。有解决方案吗?

c++ 11标准第14.3.2/1段:

非类型、非模板模板形参的模板实参必须是:

中的

—对于整型或枚举型的非类型模板形参,转换后的常量表达式(5.19)模板参数的类型;或者

—非类型模板参数的名称;或者

——一个常量表达式(5.19),用于指定具有静态存储持续时间和的对象的地址外部或内部链接或具有外部或内部链接的函数,[…]

— [...]

这意味着即使GUID本身不能用作模板参数,您也可以使用全局GUID地址作为参数,并使用指向GUID的指针作为相应的非类型参数:

template<GUID const* pGuid>
struct dll { };
template<>
struct dll<&CLSID_EH264VD>
//       ^^^^^^^^^^^^^^
//       This is a constant expression
{
    char const* filename = ""mc_dec_avc_ds.ax";
}
// ...
dll<&CLSID_EH264VD> x;

你可以试试

template<size_t B1,size_t B2,size_t B3,size_t B4,size_t B5,size_t B6,size_t B7,size_t B8,size_t B9,size_t B10>
class clsid_t{};
typedef clsid_t<0x96b9d0ed, 0x8d13, 0x4171,  0xa9, 0x83, 0xb8, 0x4d, 0x88, 0xd6, 0x27, 0xbe> CLSID_EH264VD;
template<>
struct dll<CLSID_EH264VD>
{
    char const* static name(){return "mc_dec_avc_ds.ax";};
}