解决方案调查
通过 Boost 预处理器 (BOOST_PP_REPEAT)
查看 Luc Touraille 提出的解决方案
#define INSTANTIATE(_, n, type) template class type<n>;
#define INSTANTIATE_N(n, type) BOOST_PP_REPEAT(n, INSTANTIATE, type)
INSTANTIATE_N(4, Point)
通过模板元编程
请参阅下面 phresnel 的解决方案和非常好的解释。这听起来像是更可取的方法。不幸的是,显式实例化只能在全局级别使用(语言限制),因此不能递归使用。如果实例化是隐式完成的(请参阅我的 previos 问题),则仅定义实际使用的那些符号(并因此导出到目标文件),即您需要定义类的每个符号一次(使用虚拟对象)。
这不是一个很好的方法,但它避免了预处理器解决方案的可能令人讨厌的问题(和可移植性问题)。
通过 Boost Preprocessor 处理多个参数。
最后,我还深入研究了 Boost 预处理库,并尝试扩展结果以创建表单的实例化
template class Node< int , 0 , 0 >;
template class Node< int , 1 , 0 >;
template class Node< int , 1 , 1 >;
template class Node< int , 2 , 0 >;
template class Node< int , 2 , 1 >;
template class Node< int , 2 , 2 >;
template class Node< float , 0 , 0 >;
template class Node< float , 1 , 0 >;
template class Node< float , 1 , 1 >;
template class Node< float , 2 , 0 >;
template class Node< float , 2 , 1 >;
template class Node< float , 2 , 2 >;
Node<Scalar, pdim, ldim>
所以一个具有Scalar
算术类型pdim
和整数和ldim <= pdim
另一个整数的模板。
我只能使用BOOST_PP_REPEAT
带有两个参数的模板扩展该方法,因为BOOST_PP_REPEAT
目前只能嵌套 3 层深。模板参数的两个级别和一个级别BOOST_PP_ENUM
是我使用这种技术可以达到的最大值。使用最多支持5个级别的文件迭代技术更加灵活。
我可以使用代码生成此代码
#define INTTOTYPE0 (int, (float, (double, _)))
#define NUM_TEMPLATE_ARGS 3
#define MAX_TEMPLATE_PARAM0 2
#define MAX_TEMPLATE_PARAM1(i0) 2
#define MAX_TEMPLATE_PARAM2(i0, i1) i1
#define CLASSNAME Node
#include "util/templateRecInstant.h"
问题中 Point 类的四个实例可以通过以下方式生成
#define NUM_TEMPLATE_ARGS 1
#define MAX_TEMPLATE_PARAM0 3
#define CLASSNAME Point
#include "util/templateRecInstant.h"
这两种方法都是通过文件“util/templateRecInstant.h”实现的,内容如下
#if !BOOST_PP_IS_ITERATING
#define MY_FILE "util/templateRecInstant.h"
#include <boost/preprocessor/iteration/iterate.hpp>
#include <boost/preprocessor/list/at.hpp>
#ifndef NUM_TEMPLATE_ARGS
#error need to define NUM_TEMPLATE_ARGS
#endif
#ifndef MAX_TEMPLATE_PARAM0
#error need to specify MAX_TEMPLATE_PARAM0, MAX_TEMPLATE_PARAM1, ..., up tp NUM_TEMPLATE_ARGS
#endif
#ifndef DEFAULT_INTTOTYPE
#define DEFAULT_INTTOTYPE (0, (1, (2, (3, (4, (5, (6, (7, (8, (9, _))))))))))
#endif
#ifndef INTTOTYPE0
#define INTTOTYPE0 DEFAULT_INTTOTYPE
#endif
#ifndef INTTOTYPE1
#define INTTOTYPE1 DEFAULT_INTTOTYPE
#endif
#ifndef INTTOTYPE2
#define INTTOTYPE2 DEFAULT_INTTOTYPE
#endif
#ifndef INTTOTYPE3
#define INTTOTYPE3 DEFAULT_INTTOTYPE
#endif
#if NUM_TEMPLATE_ARGS > 0
#define BOOST_PP_ITERATION_PARAMS_1 (3, (0, \
MAX_TEMPLATE_PARAM0, MY_FILE ))
#include BOOST_PP_ITERATE()
#endif
#if NUM_TEMPLATE_ARGS == 0
template class CLASSNAME< \
>;
#endif
#undef MY_FILE
#undef NUM_TEMPLATE_ARGS
#undef CLASSNAME
#undef MAX_TEMPLATE_PARAM0
#undef MAX_TEMPLATE_PARAM1
#undef MAX_TEMPLATE_PARAM2
#undef MAX_TEMPLATE_PARAM3
#undef INTTOTYPE0
#undef INTTOTYPE1
#undef INTTOTYPE2
#undef INTTOTYPE3
#elif BOOST_PP_ITERATION_DEPTH() == 1
#if NUM_TEMPLATE_ARGS > 1
#define BOOST_PP_ITERATION_PARAMS_2 (3, (0, \
MAX_TEMPLATE_PARAM1(BOOST_PP_FRAME_ITERATION(1)), \
MY_FILE))
#include BOOST_PP_ITERATE()
#endif
#if NUM_TEMPLATE_ARGS == 1
template class CLASSNAME< \
BOOST_PP_LIST_AT( INTTOTYPE0, BOOST_PP_FRAME_ITERATION(1)) \
>;
#endif
#elif BOOST_PP_ITERATION_DEPTH() == 2
#if NUM_TEMPLATE_ARGS > 2
#define BOOST_PP_ITERATION_PARAMS_3 (3, (0, \
MAX_TEMPLATE_PARAM2(BOOST_PP_FRAME_ITERATION(1) \
, BOOST_PP_FRAME_ITERATION(2) \
), \
MY_FILE))
#include BOOST_PP_ITERATE()
#endif
#if NUM_TEMPLATE_ARGS == 2
template class CLASSNAME< \
BOOST_PP_LIST_AT( INTTOTYPE0, BOOST_PP_FRAME_ITERATION(1)) \
, BOOST_PP_LIST_AT( INTTOTYPE1, BOOST_PP_FRAME_ITERATION(2)) \
>;
#endif
#elif BOOST_PP_ITERATION_DEPTH() == 3
#if NUM_TEMPLATE_ARGS > 3
#define BOOST_PP_ITERATION_PARAMS_4 (3, (0, \
MAX_TEMPLATE_PARAM3(BOOST_PP_FRAME_ITERATION(1) \
, BOOST_PP_FRAME_ITERATION(2) \
, BOOST_PP_FRAME_ITERATION(3) \
), \
MY_FILE))
#include BOOST_PP_ITERATE()
#endif
#if NUM_TEMPLATE_ARGS == 3
template class CLASSNAME< \
BOOST_PP_LIST_AT( INTTOTYPE0, BOOST_PP_FRAME_ITERATION(1)) \
, BOOST_PP_LIST_AT( INTTOTYPE1, BOOST_PP_FRAME_ITERATION(2)) \
, BOOST_PP_LIST_AT( INTTOTYPE2, BOOST_PP_FRAME_ITERATION(3)) \
>;
#endif
#elif BOOST_PP_ITERATION_DEPTH() == 4
#if NUM_TEMPLATE_ARGS == 4
template class CLASSNAME< \
BOOST_PP_LIST_AT( INTTOTYPE0, BOOST_PP_FRAME_ITERATION(1)) \
, BOOST_PP_LIST_AT( INTTOTYPE1, BOOST_PP_FRAME_ITERATION(2)) \
, BOOST_PP_LIST_AT( INTTOTYPE2, BOOST_PP_FRAME_ITERATION(3)) \
, BOOST_PP_LIST_AT( INTTOTYPE3, BOOST_PP_FRAME_ITERATION(4)) \
>;
#endif
#if NUM_TEMPLATE_ARGS > 4
#error "NUM_TEMPLATE_ARGS > 4 is not supported (limitation by boost)"
#endif
#endif
如果你偶然发现了这个问题,请随意使用上面的代码。为了生成此代码,我使用1作为参考。