C++模板 类模板成员函数类外实现 类模板分文件编写的问题与解决方法 类模板配合友元函数的类内和类外实现

1 类模板成员函数类外实现

学习目标: 能够掌握类模板中的成员函数类外实现
场景描述: 创建一个类,类中只写函数的声明,类外写实现

template<class T1, class T2>
class Person
{
public:
	Person(T1 name, T2 age);//类内声明
	void showPerson();//类内声明
	T1 mname;
	T2 mage;
};
//构造函数类外实现
template<class T1, class T2>
Person<T1, T2>::Person(T1 name, T2 age)
{
	this->mname = name;
	this->mage = age;
}
//成员函数类外实现
template<class T1, class T2>
void Person<T1, T2>::showPerson()
{
	cout << "姓名:" << this->mname << "\t年龄:" << this->mage << endl;
}

在这里插入图片描述

注意点: 构造函数/成员函数类外实现

1.构造函数前要加作用域——Person::
2.在构造函数前,声明类模板的参数,让编译器知道T1、T2的存在——template<class T1, class T2>
3.加模板的参数列表,表明Person是一个类模板——<T1, T2>,如果没有这个,Person只是一个普通类的类外实现

总结: 类模板中成员函数类外实现时,需要加上模板参数列表

2 类模板分文件编写

学习目标: 掌握类模板成员函数分文件编写产生的问题以及解决方式

问题: 类模板中成员函数创建时机是在调用阶段,导致分文件编写时链接不到

解决方法:

  • 解决方法1:直接包含.cpp源文件
  • 解决方法2:将声明和实现写到同一个文件中,并更改后缀名为.hpp,hpp是约定的名称,并不是强制

场景描述: 创建头文件写类声明,创建源文件写类实现
头文件Person.h

#pragma once
#include <iostream>
using namespace std;

template<class T1, class T2>
class Person
{
public:
	Person(T1 name, T2 age);
	void showPerson();
	T1 mname;
	T2 mage;
};

源文件Person.cpp

#pragma once
#include <iostream>
using namespace std;

template<class T1, class T2>
class Person
{
public:
	Person(T1 name, T2 age);
	void showPerson();
	T1 mname;
	T2 mage;
};

测试文件:

#include <iostream>
using namespace std;

//类模板分文件编写
#include "Person.h"
void test()
{
	Person<string, int> p1("Annas", 25);
	p1.showPerson();
}

int main()
{
	test();
	cout << endl;

	system("pause");
	return 0;
}

解释:
此时无法正常编译,因为类模板中Person构造函数和showPerson成员函数在一开始分文见编写时是不会创建的,成员函数的创建时机是在调用阶段,之前的学习笔记有讲到,导致分文件编写时链接不到。

具体来说,在测试文件中,如果包含“Person.h”这个头文件,即使编译器找到Person类,不会生成类中的Person和showPerson这两个成员函数,不会链接到Person.cpp源文件,所以不会生成这两个函数。也就是说,在一开始头文件中,类的成员函数并没有创建,导致无法链接到Person.cpp,从而不能调用类的两个成员函数。

解决办法1:包含源文件,不常用
在测试文件中,把包含的头文件“Person.h”的后缀“.h”改成“.cpp”,就可以正常调用了。相当于让编译器直接找到类的两个成员函数的具体实现。

#include "Person.cpp"

但是这种方法在实际中并不常用,通常不会让程序员直接看源码,而是看头文件。

解决办法2:将.h和.cpp的内容写到一起,并将头文件.h的后缀名改为.hpp,常用

//Person.hpp文件
#pragma once
#include <iostream>
using namespace std;

template<class T1, class T2>
class Person
{
public:
	Person(T1 name, T2 age);
	void showPerson();
	T1 mname;
	T2 mage;
};

template<class T1, class T2>
Person<T1, T2>::Person(T1 name, T2 age)
{
	this->mname = name;
	this->mage = age;
}

template<class T1, class T2>
void Person<T1, T2>::showPerson()
{
	cout << "姓名:" << this->mname << "\t年龄:" << this->mage << endl;
}

//测试文件
#include "Person.hpp"

在这里插入图片描述
注意点:

  1. .hpp文件中的类一般都是类模板
  2. .hpp是约定俗成的
  3. 直接包含**.hpp**文件

总结:
知道分文见编写出现的问题以及两种解决方法
主流的解决方式是第二种,将类模板成员函数写到一起,并将后缀名改为.hpp

3 类模板友元

学习目标: 掌握类模板配合友元函数的类内和类外实现

全局函数的实现:

  • 全局函数类内实现:直接在类内声明友元即可
  • 全局函数类外实现:需要提前让编译器知道全局函数的存在

场景描述: 创建一个类,让全局函数类内实现和类外实现,访问类的私有属性

类内实现-代码展示:

template<class T1, class T2>
class Person
{
	//全局函数类内实现 做友元 让printPerson可以访问Person的私有属性
	friend void printPerson1(Person<T1, T2> p)//Person<T1, T2>是类模板
	{
		cout << "姓名:" << p.mname << "\t年龄:" << p.mage << endl;
	}
public:
	Person(T1 name, T2 age)
	{
		this->mname = name;
		this->mage = age;
	}
private:
	T1 mname;
	T2 mage;
};

在这里插入图片描述
注意点: 全局函数在类内实现时:只需要在类内声明友元即可,加friend关键字

类外实现-代码展示:

//让编译器提前知道printPerson2中的Person是类模板
template<class T1, class T2> class Person;//类模板声明

//如果声明了函数模板,全局函数的实现可以写在后面,否则需要将实现体写在类内的前面,让编译器提前知道
template<class T1, class T2>
void printPerson2(Person<T1, T2> p)
{
	cout << "姓名:" << p.mname << "\t年龄:" << p.mage << endl;
}

template<class T1, class T2>
class Person
{
	//全局函数类外实现 类内声明
	//如果全局函数是函数模板且类外实现,需要让编译器提前知道这个函数的存在
	friend void printPerson2<>(Person<T1, T2> p);//加空模板参数列表<>
public:
	Person(T1 name, T2 age)
	{
		this->mname = name;
		this->mage = age;
	}
private:
	T1 mname;
	T2 mage;
};

在这里插入图片描述

注意点:
如果只是在类外写了全局函数的实现体,没有其他声明。此时编译器不知道printPerson3是什么类型的函数,是普通函数还是类模板的成员函数?

其实,printPerson3是普通函数,在类内仅做了声明。但在类外,由于printPerson3的前面加了函数模板声明template<class T1, class T2>,所以printPerson3在类外是一个函数模板的实现。由于printPerson3在类内、类外不一致,导致无法正常编译。因此,有如下步骤

首先,为了在类内表明printPerson3是一个函数模板的声明,需要在类内做友元时加空模板参数列表<>,即friend void printPerson3<>(Person<T1, T2> p);

其次,如果全局函数是函数模板,且类外实现,需要让编译器提前知道这个函数的存在。也就是将printPerson3的实现体放在类模板的前面。如果声明了函数模板,全局函数的实现也可以写在后面。

最后,还要声明printPerson3中的Person是一个类模板。

**总结:**建议全局函数做类内实现,用法简单,而且编译器可以直接识别

  • 0
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
面向对象程序设计课程作业 1. 请创建一个数据型为T的链表类模板List,实现以下成员函数: 1) 默认构造函数List(),将该链表初始化为一个空链表(10) 2) 拷贝构造函数List(const List& list),根据一个给定的链表构造当前链表(10) 3) 析构函数~List(),释放链表的所有节点(10) 4) Push_back(T e)函数,往链表最末尾插入一个元素为e的节点(10) 5) operator<<()友元函数,将链表的所有元素按顺序输出(10) 6) operator=()函数,实现两个链表的赋值操作(10) 7) operator+()函数,实现两个链表的连接,A=B+C(10) 2. 请编写main函数,测试该类模板的正确性: 1) 用List模板定义一个List型的模板类对象int_listB,从键盘读入m个整数,调用Push_back函数将这m个整数依次插入到该链表;(4) 2) 用List模板定义一个List型的模板类对象int_listC,从键盘读入n个整数,调用Push_back函数将这n个整数依次插入到该链表;(4) 3) 用List模板定义一个List型的模板类对象int_listA,调用List的成员函数实现A = B + C;(4) 4) 用cout直接输出int_listA的所有元素(3) 5) 用List模板定义List型的模板类对象double_listA, double_listB, double_listC,重复上述操作。(15) 3. 输入输出样例: 1) 输入样例 4 12 23 34 45 3 56 67 78 3 1.2 2.3 3.4 4 4.5 5.6 6.7 7.8 2) 输出样例 12 23 34 45 56 67 78 1.2 2.3 3.4 4.5 5.6 6.7 7.8
实验一 C++简单程序设计(2学时) 1.编程计算图形的面积。程序可以计算圆形、长方形、正方形的面积、运行时先提示用户选择图形型,然后,对圆形要求用户输入半径、对长方形要求用户输入长和宽的值,对正方形要求用户输入边长,计算出面积后将其显示出来。要求使用debug调试功能观察程序运行变量值的变化情况。 2.定义一个表示时间的结构体,可以精确的表示年、月、日、小时、、秒;提示用户输入年、月、日、小时、、秒的值,然后完整地显示出来。 实验二 函数的应用(2学时) 1.编写重载函数Max1,别求出两个整数,两个双精度数,三个整数,三个双精度数的最大值。 2.使用重载函数模板重新实现上面的函数Max1。 要求:(1)练习重载函数的使用;(2) 练习函数模板的使用。 实验三 与对象(2学时) 1.声明一个Dog,包含age、weight等属性,以及对这些属性操作的方法实现并测试这个。 2.设计并测试一个名为Rectangle的矩形,其属性为矩形的左下角和右上角两个点的坐标,有成员函数能计算矩形的面积。 3.定义一个CPU,包含等级、频率,电压等属性,并编写构造函数、析构函数,以及成员函数run、stop模拟CPU的状态。其,等级为整型,频率为单位是兆赫兹的整数,电压为浮点型。要求自己设计各个属性的标识。 4.定义一个简单的Computer,包含数据成员cpu、ram、cdrom等等,有两个成员函数run、stop。其cpu为CPU的一个对象,ram为RAM的一个对象,cdrom为CDROM的一个对象,定义并实现这个。 5.(必做)设计一个用于人事管理的People。考虑到通用性,可以只抽象出所有人员都 具有的属性:number(编号),sex(性别) ,birthday(出生日期),id(身份证号)等等。其“出生日期”定义为一个“日期”内嵌子对象。用成员函数实现对人员函数的录入和显示。要求包括:编写构造函数和析构函数、拷贝构造函数、内联成员函数的组合。 实验四 C++程序的结构(2学时) 1.编写程序,实现并测试客户机(Client)。定义字符型静态数据成员ServerName[10],保存其服务器名称;整型静态数据成员ClientNum,记录定义的客户数量;定义静态函数ChangeServerName()改变服务器名称。在头文件client.h定义,在文件client.cpp实现,在文件test.cpp测试这个,观察相应的成员变量的取值的变化情况。 2、在实验三题目5编写的人员设计适当的方法实现数据的共享性,并采用多文件结构实现程序。 3.(选做)定义X、Y、Z,函数h(X *),满足:X有私有成员i,Y的成员函数g(X *)是X的友元函数实现对X的成员i加1,Z是X的友元,其成员函数f(X *)实现对X的成员i加5,函数h(X *)是X的友元函数实现对X的成员i加10。在一个文件声明,在一个文件实现,在另一个文件测试。 实验五 继承和派生(2学时) 1.设计并定义一个交通工具,并通过该派生出新的编写程序定义这些并测试它; 2.(选做)声明一个基Shape,在此基础上派生Rectangle和Circle,二者都有GetArea()函数计算对象的面积,编写程序测试。 3.(选做)声明一个哺乳动物Mammal,再由此派生出狗Dog,定义一个Dog的对象,观察基与派生的构造函数和析构函数的调用顺序。 4.完善实验四第2题的程序,具体要求如下: 任务1:从people(人员)派生出student(学生),添加属性:学校、学院、专业、班号、入学成绩,设计相应的成员函数(构造函数,拷贝构造函数,录入函数,显示函数)。 任务2:从people派生出teacher(教师),添加属性:职务,部门,职称。并设计相应的成员函数。 任务3:从student派生出graduate(研究生),添加属性:导师,研究方向。并设计相应的成员函数。 任务4:编写程序来测试这个。 实验六 小型学生管理系统的设计与实现1(2学时) 设计学生、课程、成绩及相应的成员函数。要求能实现学生的信息、课程信息和成绩信息的增加、修改、删除与查询。 实验七 多态性(2学时) 1.定义Point,有坐标X和Y两个成员变量,对Point重载++和――运算符,实现对坐标值的改变。 2.定义一个车(Vehicle),有run,stop等成员函数,由此派生出自行车(bicycle)、汽车(car),由bicycle和car派生出摩托车(motocar),它们都包含run,stop等成员函数编写相应的虚函数并测试

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值