对于一个应用程序,我需要确保一个类型(Instance
)已经注册到另一个类型(System
),然后再与它一起使用。
通过创建Register
类的全局实例进行注册:
// Instance.cpp
#include "Instance.h"
#include "System.h"
#include "Register.h"
static Register<Instance, System> instanceRegistration;
注册后,Instance
可与System
结合使用,在此简化示例中,您可以调用runme
函数:
// main.cpp
#include "Instance.h"
#include "System.h"
template <typename T, typename T_System>
void runme() {
// TODO: error if type T was not registered for type T_System.
// ...
// Code that fails if no instance of Register<Instance, System> exists
// ...
}
int main() {
runme<Instance, System>();
}
使用未注册的模板参数组合调用runme
应该会失败(在这种情况下,真正的代码会出现段错误。)
目前,runme
针对在bool
的构造函数中设置的全局Register
进行了运行时检查。如果全局bool
为false
,则会引发异常。这很好。
但是,我想将错误检查移到编译时,因为从理论上讲,链接器应该能够确定是否存在Register
类的全局实例。
如果在没有runme<Instance, System>()
类型的全局变量的情况下调用Register<Instance, System>
,是否有某种方法可以生成链接器错误?
其他头文件如下:
// Instance.h
class Instance { };
// System.h
class System { };
// Register.h
template <typename T, typename T_System>
class Register {
public:
Register() {
// Some side effects...
}
};
答案 0 :(得分:1)
这并不完美,但它可以帮助你完成大部分工作。首先,声明一个名为(例如)EnsureRegistered()
的未实现函数,该函数与Register
在相同的参数上进行模板化;也许在Register.h:
template<typename T, typename T_System>
void EnsureRegistered(); // not implemented here!
现在,无论何时创建Register
,都要提供EnsureRegistered()
的显式特化(可能使用预处理器宏来自动执行此操作):
Register<int, int> g_Register_i_i;
template<>
void EnsureRegistered<int, int>() {}
最后,在runme()
中调用该函数(或者只是获取其地址并将其分配给某些内容,如果性能至关重要):
template<typename T, typename T_System>
void runme()
{
EnsureRegistered<T, T_System>();
// ...
}
现在,runme<int, int>()
将构建,但runme<int, double>()
会出现链接错误。