免责声明 - 老实说我不确定如何构建这个问题或提供适当的背景,如果有这方面经验的人可以推荐我需要提供哪些额外信息来澄清上下文,我将不胜感激。而且如果我需要清理下面的代码部分以使其更清晰,我将在收到评论时这样做!
但无论如何这里 -
我正在尝试使用可变参数模板,但每次我编译我的代码(它在公司代码库上)时,编译器(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))
答案 0 :(得分:1)
模板仅在使用时实例化 。将定义放在实现中而不在该文件中使用它们将导致无实例化。然后,当您尝试在其他地方使用它时,它将找不到实例化的实现,因为它在当前翻译单元无法看到的实现文件中。您应该将实现保留在标头中。 See this question
如果您有NpBaseTest
的定义,并且您移动了标题中的所有代码(我假设为TestFactory.h),那么您的示例工作正常。以下是如何在示例中使用代码的示例。请注意,模板的编写方式,它们只接受指向返回NpBaseTest*
的函数的指针。
// main.cpp
#include "TestFactory.h"
#include <iostream>
NpBaseTest* test_callback(int x, int y)
{
std::cout << "Called test_callback(" << x << ", " << y << ")\n";
return nullptr;
}
int main()
{
// This instantiates TestFactory::RegisterTest<int, int>
TestFactory::RegisterTest<int, int>("my test", &test_callback);
// This instantiates TestFactory::CreateTest<int, int>
NpBaseTest * result = TestFactory::CreateTest<int, int>("my test", 5, 10);
return 0;
}
我为clariy明确写了模板参数。在您的情况下,编译器将能够推导出这些参数,并且示例大大简化。您可以简单地调用模板化方法而不使用任何参数,它们将从参数中推断出来。
// main.cpp
#include "TestFactory.h"
#include <iostream>
NpBaseTest* test_callback(int x, int y)
{
std::cout << "Called test_callback(" << x << ", " << y << ")\n";
return nullptr;
}
int main()
{
// This instantiates TestFactory::RegisterTest<int, int>
TestFactory::RegisterTest("my test", &test_callback);
// This instantiates TestFactory::CreateTest<int, int>
NpBaseTest * result = TestFactory::CreateTest("my test", 5, 10);
return 0;
}
那就是它。如果尝试此示例,您会看到现在生成了TestFactory::RegisterTest<int, int>
和TestFactory::CreateTest<int, int>
的符号。