C++中类模板的定义和使用

引言

类模板就是一个模板,但是数据可以适用多种类型。类模板使用时需要模板的特例化,就变成了模板类。
本文只要是记录一下模板的使用。同时对于引用和右值引用传参做一下记录。

类模板

声明和定义

类模板的声明和定义一般都是放在同一个文件中,将其文件后缀名改为hpp。

有问有答

为什么要将类模板的声明和定义放在同一个函数

  1. 类模板在编译的时候进行模板实例化,若将声明与定义分别放在两个文件中,可能会出现找不到模板的定义,导致编译错误。
  2. 且由于模板实例化发生在编译时,而不是链接时,因此不需要为模板生成单独的目标文件。如果声明和定义分开,编译器可能会尝试为模板的定义生成目标文件,这会导致链接错误。

为什么头文件为hpp

hpp文件允许函数的声明和定义放在一个文件中。且声明为hpp后,一看头文件的名字就知道声明与定义在一个文件中。

示例

下面是一个类模板的示例。
Counter.hpp

#pragma once
template<typename T>
class MyClass
{
public:
	MyClass();
	~MyClass();
	void setValue(T& val);
	T* getValue()const;
	T getData()const;
private:
	T* m_ptr;
};

template<typename T>
MyClass<T>::MyClass() :m_ptr(nullptr)
{
	m_ptr = new T;
}

template<typename T>
MyClass<T>::~MyClass()
{
	delete m_ptr;
	m_ptr = nullptr;
}

template<typename T>
void MyClass<T>::setValue(T& val) {
	*m_ptr = val;
}

template<typename T>
T* MyClass<T>::getValue()const {
	return m_ptr;
}

template<typename T>
inline T MyClass<T>::getData() const
{
	return T(*m_ptr);
}

main.cpp

#include <iostream>
#include "Counter.hpp"

using namespace std;


int main(int argc,char *argv[]) {
    MyClass<int> obj;
    int a = 45;
    obj.setValue(a);//右值引用可以使用右值,或者将左值转换为右值来传递参数,左值引用只能使用左值传递参数
    cout << "设置后其地址为:" << obj.getValue() << endl;
    cout << "设置后其值为:" << obj.getData() << endl;
	return 0;
}

运行结果

在这里插入图片描述

注意

参数传递

  1. 引用
    上面的示例中,void setValue(T& val);参数为左值引用,此时在调用函数setValue的时候,传入参数时需要传入一个变量,而不能传入一个数值。
   obj.setValue(3);

这样会编译不通过。左值引用为变量的别名。
2. 右值引用
当将上面的函数改为void setValue(T&& val);其参数为右值引用类型,这时调用函数setValue时,传参需要使用move移动语义。

void setValue(T&& val);

template<typename T>
void MyClass<T>::setValue(T&& val) {
	*m_ptr = val;
}

使用的时候,可以使用move传参。

#include <iostream>
#include "Counter.hpp"

using namespace std;


int main(int argc,char *argv[]) {
    MyClass<int> obj;
    int a = 45;
    obj.setValue(move(a));//右值引用可以使用右值,或者将左值转换为右值来传递参数,左值引用只能使用左值传递参数
    cout << "设置后其地址为:" << obj.getValue() << endl;
    cout << "设置后其值为:" << obj.getData() << endl;
	return 0;
}

当然,也可以直接使用一个右值来传参。下面数值3就是一个右值。

#include <iostream>
#include "Counter.hpp"

using namespace std;


int main(int argc,char *argv[]) {
    MyClass<int> obj;
    obj.setValue(3);//右值引用可以使用右值,或者将左值转换为右值来传递参数,左值引用只能使用左值传递参数
    cout << "设置后其地址为:" << obj.getValue() << endl;
    cout << "设置后其值为:" << obj.getData() << endl;
	return 0;
}

ref

ref用于函数式传参,可以用于bind对象绑定器的传参,也可以用于thread线程传参。在对象绑定器中所指向的函数形参为引用时,若直接使用变量来传参,就会出现拷贝,而使用ref包装一下变量,就使用变量的引用,可以改变变量自身的值,不会出现拷贝。
详情可以参考:
ref的初步讲述

  • 14
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

肩上风骋

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值