类模板成员函数的创建时机是在调用阶段,导致分文件编写时链接不到。
#pragma once // 防止头文件重复包含
#include<iostream>
#include<string>
using namespace std;
// 声明
template<class T1,class T2>
class Person
{
public:
Person(T1 name,T2 age);
void showPerson();
T1 m_Name;
T2 m_Age;
};
#include "person.h"
template<class T1,class T2>
Person<T1,T2>::Person(T1 name,T2 age)
{
this->m_Name = name;
this->m_Age = age;
}
template<class T1,class T2>
void Person<T1,T2>::showPerson()
{
cout<<"姓名: "<<this->m_Name<<"年龄: "<<this->m_Age<<endl;
}
#include<iostream>
#include "person.h"
using namespace std;
void test01()
{
Person<string,int> P("Jie",18);
P.showPerson();
}
int main()
{
test01();
return 0;
}
编译报错。
因为类模板成员函数一开始是不会创建的,
当你去包含.h的时候,相当于只是让编译器看到了person.h(声明代码)。编译器没有看到person.cpp中的实现构造函数和 void showPerson()的代码。所以链接阶段失败。
1:直接包含.cpp源文件
2.将声明和实现写到同一个文件中,并更改后缀名为.hpp,.hpp是约定的名称。
把 .h改为.cpp时,相当于让编译器看到实现的代码。
#include<iostream>
#include "person.cpp"
using namespace std;
void test01()
{
Person<string,int> P("Jie",18);
P.showPerson();
}
int main()
{
test01();
return 0;
}
此时编译器可以成功编译了。
但是一般不用这种方法,因为一般是不会直接导入cpp源码的。
include/person.hpp
#pragma once // 防止头文件重复包含
#include<iostream>
#include<string>
using namespace std;
// 声明
template<class T1,class T2>
class Person
{
public:
Person(T1 name,T2 age);
void showPerson();
T1 m_Name;
T2 m_Age;
};
// 实现
template<class T1,class T2>
Person<T1,T2>::Person(T1 name,T2 age)
{
this->m_Name = name;
this->m_Age = age;
}
template<class T1,class T2>
void Person<T1,T2>::showPerson()
{
cout<<"姓名: "<<this->m_Name<<"年龄: "<<this->m_Age<<endl;
}
mian.cpp
#include<iostream>
#include "person.hpp"
using namespace std;
void test01()
{
Person<string,int> P("Jie",18);
P.showPerson();
}
int main()
{
test01();
return 0;
}
编译:
g++ main.cpp -Iinclude
运行:
./a.out