从单独的文件转发模板函数的声明
Forward declaration of a template function from a seperate file
其他.cpp
#include <iostream>
#include <string>
using std::cout;
template <typename T>
std::ostream & cprint(T &t, std::string msg = "Container", std::ostream & stream = cout){
stream << msg << "t{ ";
for(typename T::iterator it = t.begin(); it != t.end(); it++)
stream << *it << " ";
stream << "}n";
return stream;
}
我在一个单独的文件中有一个模板函数。我正试图在main.cpp文件中转发声明它。
extern template std::ostream & cprint<std::vector<int>>
(T &t, std::string msg = "Container", std::ostream & stream = cout);
我试过上面的东西,但对我不起作用。
编辑:假设other.cpp具有如下基本功能,
template <typename T>
void func(T x){
cout << x << endl;
}
如何在main.cpp中实例化这个函数?
请记住,您所询问的(通常)不是使用模板的正确方法
您应该做的是在头中声明模板,并去掉extern
声明和显式专门化
将模板放入.cpp
通常不是很方便,因为它们要求您指定希望它们在同一.cpp
中工作的所有类型。
但是,另一方面,它提高了编译速度,并且生成的二进制文件较小的可能性很小。
如何修复代码:
我对您的代码做了一些更改,使其正常工作。
我也做了一些小的改进。如果你觉得它们令人困惑,请在评论中提问。
// 1.cpp
#include <iostream>
#include <string>
#include <vector>
template <typename T>
std::ostream &cprint(const T &t, std::string msg = "Container", std::ostream &stream = std::cout)
{
stream << msg << " { ";
for(typename T::const_iterator it = t.cbegin(); it != t.cend(); it++)
stream << *it << " ";
stream << "}n";
return stream;
}
template std::ostream &cprint(const std::vector<int> &, std::string, std::ostream &);
// 2.cpp
#include <string>
#include <iostream>
#include <vector>
template <typename T> std::ostream &cprint(const T &, std::string = "Container", std::ostream & = std::cout);
int main()
{
std::vector<int> v{1,2,3};
cprint(v);
}
重要部件如下:
显式实例化。它必须与模板定义位于同一个文件中
template std::ostream &cprint(const std::vector<int> &, std::string, std::ostream &);
声明。它必须位于要使用模板的文件中。
template <typename T> std::ostream &cprint(const T &, std::string = "Container", std::ostream & = std::cout);
请注意,如果编译器无法推导出模板参数,则可以在显式实例化和外部声明中手动指定模板参数。
用一个被编辑成问题的虚拟模板做同样的事情留给读者做练习。
除此之外,您还应该阅读:使用外部模板(C++11)
要推迟使用extern
的实例化,您必须与函数签名完全匹配,因此除非T
定义为std::vector<int>
或其别名(与您的模板声明匹配),否则以下内容将不起作用
extern template std::ostream & cprint<std::vector<int>>
(T &t, std::string msg = "Container", std::ostream & stream = cout);
要修复,
extern template std::ostream& cprint<std::vector<int>>
(std::vector<int>&, std::string msg = "Container", std::ostream & stream = cout);
点击此处查看
更具体地说,
在Header.hpp文件中,我有它的声明和延迟:
template <typename T>
std::ostream & cprint(T &t, std::string msg = "Container", std::ostream & stream = cout){
stream << msg << "t{ ";
for(typename T::iterator it = t.begin(); it != t.end(); it++)
stream << *it << " ";
stream << "}n";
return stream;
}
//defer instantiation
extern template std::ostream& cprint<std::vector<int>>
(std::vector<int>&, std::string msg = "Container", std::ostream & stream = cout);
现在,在main.cpp文件中,我有:
int main(){
std::vector<int> v{1, 2, 3, 4, 5};
cprint(v);
}
在Header.cpp文件中,我有它的实例化:
template std::ostream& cprint<std::vector<int>>
(std::vector<int>&, std::string msg = "Container", std::ostream & stream = cout);
如图
=====================
编辑:关于您最近的编辑:
编辑:假设other.cpp有如下基本功能,
template <typename T> void func(T x){ cout << x << endl; }
如何在main.cpp中实例化这个函数?
如果翻译单元main.cpp看到函数模板func
的声明(来自包含的标头,或者您(re-(forward-))在main.cpp中声明了它,它将编译为调用功能模板专用化1,但是链接器将无法找到函数模板专用化,除非它看到功能模板专用化的定义在(另一个)翻译单元中实例化。
注意:">函数模板专用化"是从函数模板实例化的函数。参见温度fct/1
- Visual Studio中的函数声明和函数定义问题
- 为什么函数声明中允许 const?
- 如果 x.h 仅由函数声明组成,为什么有必要在 x 中包含 x.h.cpp
- * 和 ** 在 C++ 函数声明中是什么意思?
- 构造函数/函数声明参数列表中的统一初始化
- 在将函数声明为友元时,尖括号的含义是什么?
- 为什么转换函数声明不需要至少一个定义类型说明符
- 如何正确编写指针函数声明?
- 在"template"和函数声明之间使用:template<typename trait> using tr = base_trait<trait> void fn(tr::t
- 为什么要将函数声明和定义放在单独的文件中
- 为什么系统日志有两个不同的函数声明?
- 我如何获取数组的大小,以便我可以从函数声明所述数组
- 使用 enable_if 在按值传递与按引用传递之间更改函数声明
- JavaScript 中的一等函数和 C++ 中的函数声明
- C++ 通过函数声明后初始化向量
- VS2017 #error: : snprintf 的宏定义与标准库函数声明冲突
- C++ 17 个友元函数声明和内联命名空间
- MSVC 2017 - 错误 - 如何将模板类 X 的模板成员函数声明为嵌套类 X::Y 的好友
- 将派生类的构造函数声明为父类的友元
- 用于从 ANSI 字符串转换为 std::basic_string <TCHAR>的正确函数声明