一.使用常规分文件创建方法实现类模板
类模板成员函数的创建时机是在调用阶段,导致分文件编写时链接不到。
include文件下的person.h
cpp
#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;
};
person.cpp
cpp
#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;
}
mian.cpp
cpp
#include<iostream>
#include "person.h"
using namespace std;
void test01()
{
Person<string,int> P("Jie",18);
P.showPerson();
}
int main()
{
test01();
return 0;
}
编译报错。
cpp
因为类模板成员函数一开始是不会创建的,
当你去包含.h的时候,相当于只是让编译器看到了person.h(声明代码)。编译器没有看到person.cpp中的实现构造函数和 void showPerson()的代码。所以链接阶段失败。
二.解决方法
cpp
1:直接包含.cpp源文件
2.将声明和实现写到同一个文件中,并更改后缀名为.hpp,.hpp是约定的名称。
2.1 直接包含.cpp源文件
把 .h改为.cpp时,相当于让编译器看到实现的代码。
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源码的。
2.2 将.h和.cpp中的声明和实现内容写到一起,将后缀名改为.hpp文件
include/person.hpp
cpp
#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
cpp
#include<iostream>
#include "person.hpp"
using namespace std;
void test01()
{
Person<string,int> P("Jie",18);
P.showPerson();
}
int main()
{
test01();
return 0;
}
编译:
bash
g++ main.cpp -Iinclude
运行:
bash
./a.out