【发布时间】:2017-02-09 21:36:05
【问题描述】:
免责声明 - 老实说,我不确定如何提出这个问题或提供适当的上下文,如果遇到此问题的人能推荐我需要提供哪些额外信息来澄清上下文,我将不胜感激。而且,如果我需要清理下面的代码部分以使其更清晰,我会在收到 cmets 时这样做!
但不管怎样-
我正在尝试使用可变参数模板,但每次我编译我的代码(它在公司代码库上)时,编译器(gcc 4.8.4 - C++11)似乎都会跳过所有可变参数代码部分 -
TestFactory.cpp
/*
* Use this design since Variadic Templates are not available in C++11
* A MapHolder allows us to create a map of variadic functions.
*/
template <class... Args>
struct MapHolder
{
static std::map<std::string, NpBaseTest*(*)(Args...)> CallbackMap;
};
template <class... Args>
std::map<std::string, NpBaseTest *(*)(Args...)> MapHolder<Args...>::CallbackMap;
class TestFactory
{
public:
template <class... Args>
static void RegisterTest(std::string name, NpBaseTest *(*callback)(Args...));
template <class... Args>
static NpBaseTest *CreateTest(const std::string &name, Args &&... args);
};
TestFactory.cpp
template <class... Args>
void TestFactory::RegisterTest(std::string name, NpBaseTest *(*callback)(Args...))
{
MapHolder<Args...>::CallbackMap[name] = callback;
}
template <class... Args>
NpBaseTest *TestFactory::CreateTest(const std::string &name, Args &&... args)
{
return (MapHolder<Args...>::CallbackMap[name])(std::forward<Args>(args)...);
}
调用文件 -
void np_test_mgr_print()
{
const char *s = "cavpkotest";
std::string str(s);
TestFactory::RegisterTest(str.c_str(), &CavPkoTest::create);
NpBaseTest *o1{TestFactory::CreateTest<uint16_t>(str.c_str(), 1)};
/* Irrelevant code section */
NpTestMgr::get_instance().insert(o1);
NpTestMgr::get_instance().submit();
}
}
当我编译这个(gcc 4.8.4)时,目标文件 TestFactory.o 是空的。如果我在我们的代码库(gcc 4.4.6)之外执行此操作,代码将被编译并输出 -
[common]$ nm TestFactory.o | c++filt $1 | grep CreateTest
34:0000000000401d6a W NpBaseTest* TestFactory::CreateTest<unsigned short>(std::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, unsigned short&&)
[common]$ nm TestFactory.o | c++filt $1 | grep RegisterTest
35:0000000000401d40 W void TestFactory::RegisterTest<unsigned short>(std::basic_string<char, std::char_traits<char>, std::allocator<char> >, NpBaseTest* (*)(unsigned short))
【问题讨论】:
-
c-标签是有原因的,还是您只是觉得它美观? -
模板仅在使用时进行实例化。将定义放在实现中而不使用它们在该文件中将导致没有实例化。然后,当您尝试在其他地方使用它时,它将找不到要实例化的实现,因为它位于当前翻译单元无法看到的实现文件中。您应该将实现留在标题中。 See this question.
-
@FrançoisAndrieux 我会试一试并提供更新!感谢您的链接!
-
@FrançoisAndrieux 因为这是一个可变参数模板,我该如何实例化它?我可以采用可变参数和类型。如果我指定类型,我是否没有违背使用可变参数模板的目的?
-
@nitimalh 我假设你打算在某个时候使用你的可变参数模板函数,在那里你会知道你需要什么类型(即使这些类型本身就是模板参数)。这是您将实例化的点。只有当您使用模板时,才会按照设计进行实例化。我觉得您担心符号未定义。不用担心,只要您实际使用模板,它们就会被定义,只要您的实现是可访问的(如果它在您的标题中,它将是可访问的)。
标签: c++ c++11 gcc compilation