第十二站(20天):C++泛型编程

模板

C++提供了模板(template)编程的概念。所谓模板,实际上是建立一个通用函数或类,
类内部的类型和函数的形参类型不具体指定 ,用一个虚拟的类型来代表。这种通用的方式称
为模板。 模板是泛型编程的基础, 泛型编程即以一种独立于任何特定类型的方式编写代码
如: vector

函数模板(可以嵌套使用)

模板函数就是函数模板的自动实现

功能:
使用函数体相同的函数都可以用这个模板替代
实现多个函数用来返回两个数的最大值,要求能支持 char 类型、int 类型、double
template <typename T>
如:
        template <typename T>//这里的类属参数 T在下面函数体的 参数中必须出现一次
        T Max(T a, T b) {
            return a > b ? a : b;
        }
形式:
cout << Max(a, b) << endl;//编译器自动推导类型
cout << Max<int>(a,b)<<endl;//显示类型调用
#include <iostream>
using namespace std;

template <typename T>
T Max(T a, T b) {
	return a > b ? a : b;
}
int main(void) {
	int a = 4;
	int b = 5;
	float x = 1.2f;
	float y = 2.6f;
	cout << Max(a, b) << endl;//编译器自动推导类型
    cout << Max<int>(a,b)<<endl;//显示类型调用
	cout << Max(x, y) << endl;
	return 0;
}

函数模板和函数重载

//当函数模板和普通函数都符合调用时,优先选择普通函数
//Max(a,b);
//如果 显式指定类型 的使用函数模板,则使用<> 类型列表
//Max<>(a, b);
//如果函数模板会产生更好的匹配,使用函数模板

 函数模板的调用机制

函数模板在调用的时候,会根据调用者的数据类型,创建一个匹配的函数

类模板

类模板和函数模板一致

在类使用数据类型的地方都可以用模板的类属参数进行替换

(成员数据的数据类型,构造函数的参数类型,成员函数的返回值类型)

注:在模板类定义对象的时候必须指定为显示指定类型<>

当为单个参数时:

#include <iostream>
using namespace std;
template <typename T>
class A {
public:
	A(T a) {
		this->a = a;
	}
	T getA() {
		return a;
	}
private:
	T a;
};

int main(void) {
	A<int> aa(666);
	cout << aa.getA() << endl;

	return 0;
}

当为多个参数时

#include <iostream>
using namespace std;
template <typename T1,typename T2>
class A {
private:
	T1 a;
	T2 b;
public:
	A(T1 a ,T2 b) {
		this->a = a;
		this->b = b;
	}
	T1 getA() {
		return a;
	}
	T2 getB() {
		return b;
	}
};

int main(void) {
	A<int, double> aa(12,4.5);
	cout << aa.getA() << endl;
	cout << aa.getB() << endl;
	return 0;
}

当类作为函数参数的时候

 double add(A<int, double>& a) {
    double sum = 0;
    sum = a.getA() + a.getB();
    return sum;
}

当子类继承父类(父类为模板类时)

 class B :public A<int, double> {
    B(int a, double b):A(a,b) {}
};

当子类是模板类,父类是一般类时 

 和普通继承一样

#include <iostream>
using namespace std;

class B {
protected:
	int  a;
	double  b;
public:
	B(int a=1, double b=1.2) {
		this->a = a;
		this->b = b;
	}
};

template <typename T1, typename T2>
class A :public B{
public:
	A();
	A(T1 a, T2 b) :B(a, b) {}
	T1 getA() {
		return a;
	}
	T2 getB() {
		return b;
	}
};

double add(A<int, double>& a) {
	double sum = 0;
	sum = a.getA() + a.getB();
	return sum;
}
int main(void) {
	A<int ,double> aa(12,12.2);
	cout << aa.getA() << endl;
	cout << aa.getB() << endl;
	return 0;
}

当子类和父类都是模板类

和普通继承一样

#include <iostream>
using namespace std;

template <typename T1, typename T2>
class B {
public:
	B(T1 a, T2 b){
		this->a = a;
		this->b = b;
	}
	T1 getA() {
		return a;
	}
	T2 getB() {
		return b;
	}
protected:
	T1 a;
	T2 b;
};

template <typename T1, typename T2>
class A :public B<int, double>{
public:
	A();
	A(T1 a, T2 b) :B(a, b) {}
	T1 getA() {
		return a;
	}
	T2 getB() {
		return b;
	}
};

double add(A<int, double>& a) {
	double sum = 0;
	sum = a.getA() + a.getB();
	return sum;
}
int main(void) {
	A<int, double> aa1(10, 10.2);
	cout << aa1.getA() << endl;
	cout << aa1.getB() << endl;
	A<int ,double> aa(12,12.2);
	cout << aa.getA() << endl;
	cout << aa.getB() << endl;

	return 0;
}

类模板和友元函数 

 

(1) 类内部声明友元函数,必须写成一下形式
template < typename T>
friend A<T> addA (A<T> &a, A<T> &b);
(2) 友元函数实现 必须写成
template < typename T>
A<T> add(A<T> &a, A<T> &b) {
        //......
}
(3) 友元函数调用 必须写成
A< int > c4 = addA <int> (c1, c2);

 类模板的封装

这里仅分为两个文件(.h和.cpp+main.cpp)

函数前声明 template
< 类型形式参数表 >
类的成员函数前的 类限定域说明必须要带上虚拟参数列表
template<typename T1>
A<T1>::A(T1 a)
{
    this->a = a;
}
返回的变量是模板类的对象时必须带上虚拟参数列表
T1 A<T1>::getA() {
    return a;
}
成员函数参数中出现模板类的对象时必须带上虚拟参数列表(也可以不带)
A<T1> A<T1>::operator+( const A<T1> & other)
{
    A<T1> tmp = this->a + other.a;
    return tmp.a;
}
成员函数内部没有限定( 可带可以不带 )
A<T1> A<T1>::operator+( const A<T1> & other)
{
  
 A<T1> tmp = this->a + other.a;
    return tmp.a;
}
.h
#include <iostream>
using namespace std;

template <typename T1>
class A {
public:
	A(T1 a);
	T1 getA();
	A operator+(const A& other);
	void print();
private:
	T1 a;
};

.cpp
#include <iostream>
using namespace std;
#include "A.h"

template<typename T1>
A<T1>::A(T1 a)
{
	this->a = a;
}
template <typename T1>
T1 A<T1>::getA() {
	return a;
}
template<typename T1>
A<T1> A<T1>::operator+(const A& other)
{
	A tmp = this->a + other.a;
	return tmp.a;
}

template<typename T1>
void A<T1>::print()
{
	cout << this->a << endl;
}
int main(void) {
	A<int> a(12);
	A<int> b(11);
	A<int> c = a + b;
	c.print();
	return 0;
}

 这里仅分为三个文件(.h 和  .cpp  和  main.cpp)

如果分为这三个文件,那么只包含.h头文件就会报错C++ 类模板“无法解析的外部符号

在封装有三个文件的时候mian.cpp中就不用包含.h头文件了,仅包含.cpp文件即可

#include <iostream>
using namespace std;

//#include "A.h"//同时包含两个或者仅包含.h都会报错
#include "A.cpp"

int main(void) {
    A<int> a(12);
    A<int> b(11);
    A<int> c = a + b;
    c.print();
    return 0;
}

 类模板的静态成员数据

  从类模板实例化的每个模板类有自己的类模板数据成员,该 模板类的所有对象共享
一个 static 数据成员
和非模板类的 static 数据成员一样, 模板类的 static 数据成员也应该在文件范围定
义和初始化
  static 数据成员也可以使用虚拟类型参数 T

public:

        static T1 count;

//初始化静态成员数据
template<typename T1>
T1 A<T1>::count = 66;

//通过对象a对静态count赋值后
cout << a.count << endl;
a.count = 888;
cout << b.count << endl;
b.count = 1000;
cout << a.count << endl; 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/338060.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

JavaWeb-Listener

一、概念 Listener表示监听器&#xff0c;是JavaWeb三大组件&#xff08;Servlet&#xff0c;Filter&#xff0c;Listener&#xff09;之一&#xff0c;监听器的监听对象可以是application, session, request三个对象&#xff0c;监听的事件是这些对象的创建或销毁&#xff0c…

虚拟机将1.15版本的nginx推送到阿里云镜像仓库

1、docker images 2、docker login --usernamealiyun7279061146 registry.cn-shenzhen.aliyuncs.com 3、docker tag 53f3fd8007f7 registry.cn-shenzhen.aliyuncs.com/zhouwb/zhou:1.15 docker push registry.cn-shenzhen.aliyuncs.com/zhouwb/zhou:1.15

Linux第33步_TF-A移植的第1步_创建新的设备树

TF-A移植第1步就是创建新的设备树&#xff0c;并命名为“stm32mp157d-atk”。 和“TF-A移植”有关的知识点&#xff1a; 1)设备树英文名字叫做Device tree&#xff0c;用来描述板子硬件信息的&#xff0c;比如开发板上的 CPU有几个核 、每个CPU核主频是多少&#xff0c;IIC、…

线性代数:逆、转置、分块、多项式 矩阵公式总结

目录 逆矩阵、转置矩阵重要公式 公式 证明 矩阵分块 基本运算 分块的逆&#xff08;主副对角线分块对角阵的逆、主副对角线上下三角分块对角阵的逆&#xff09; 例 矩阵多项式 例 克拉默法则及逆矩阵求方程组 逆矩阵、转置矩阵重要公式 公式 证明 矩阵分块 基本运…

科技护航 智慧军休打通医养结合最后一公里

“小度小度&#xff0c;请帮我打电话给医生。” “好的&#xff0c;马上呼叫植物路军休所医生。” 2023年9月25日&#xff0c;常年独居、家住广西南宁市植物路军休所的军休干部程老&#xff0c;半夜突发疾病&#xff0c;让他想不到的是&#xff0c;这个常年伴他左右的“小度”…

刷题日记-139. 单词拆分

这是一道动态规划题目&#xff0c;要求判断给出的字符串s能否被wordDict字符串列表中的字符串组成。 这段代码是一个解决单词拆分问题的函数 wordBreak&#xff0c;其作用是判断字符串 s 是否可以被拆分为由字典 wordDict 中的单词组成。 我们要通过构建一个布尔值的向量 dp&…

【Godot4自学手册】第一节配置Godot运行环境

各位同学大家好&#xff01;我是相信神话&#xff0c;从今天开始&#xff0c;我开始自学2D游戏开发&#xff0c;用到的是Godot4。我准备用视频记录整个开发过程&#xff0c;为自学2D开发的同学趟趟路。让我们开始吧。 首先介绍一下Godot是什么东西&#xff0c;在2D游戏开发中是…

DolphinDB学习(0):DolphinDB基本概述

DolphinDB的学习难度不小&#xff0c;主要是写法比较多&#xff0c;官方示例是一次性给一大堆代码&#xff0c;在没有成体系的学习基础的前提下&#xff0c;总有种力不从心的感觉&#xff0c;所以博主汇总这一个系列的文章&#xff0c;尝试从最简单的基础常规操作开始&#xff…

springboot113健身房管理系统

简介 【毕设源码推荐 javaweb 项目】基于springbootvue 的健身房管理系统 适用于计算机类毕业设计&#xff0c;课程设计参考与学习用途。仅供学习参考&#xff0c; 不得用于商业或者非法用途&#xff0c;否则&#xff0c;一切后果请用户自负。 看运行截图看 第五章 第四章 获取…

力扣刷MySQL-第七弹(详细讲解)

&#x1f389;欢迎您来到我的MySQL基础复习专栏 ☆* o(≧▽≦)o *☆哈喽~我是小小恶斯法克&#x1f379; ✨博客主页&#xff1a;小小恶斯法克的博客 &#x1f388;该系列文章专栏&#xff1a;力扣刷题讲解-MySQL &#x1f379;文章作者技术和水平很有限&#xff0c;如果文中出…

一文详解 Berachain 测试网:全面介绍与教程,bitget wallet教程

什么是Berachain&#xff1f; Berachain&#xff08;web3.bitget.com/zh-CN/assets/berachain-wallet&#xff09;是一种尖端区块链技术&#xff0c;使用 Cosmos SDK 构建的 Layer-1&#xff0c;兼容以太坊虚拟机&#xff08;EVM&#xff09;。它基于一种独特的概念&#xff0c…

Summerize for Bioinformatics with ChatGPT

目录 Basic Introduction Historical Events Bioinformatics vs. Computational Biology Levels of Bioinfo & CompBio Molecular Biology Primer Homework 1 A brief Introduction to DNA & RNA A brief Introduction to Protein Sequencing Technologies S…

解决git在使用代理时进行HTTPS操作时的无法连接github的解决办法

问题如下&#xff1a; 在使用git clone 无法连接github的443端口。 解决问题&#xff1a; 首先查看自己的代理服务器使用的端口号&#xff08;我这里的端口号是7890。&#xff09;使用指令 git config --global https.proxy 127.0.0.1:7890这条命令是在配置 Git 使用代理服务…

【数据结构与算法】归并排序详解:归并排序算法,归并排序非递归实现

一、归并排序 归并排序是一种经典的排序算法&#xff0c;它使用了分治法的思想。下面是归并排序的算法思想&#xff1a; 递归地将数组划分成较小的子数组&#xff0c;直到每个子数组的长度为1或者0。将相邻的子数组合并&#xff0c;形成更大的已排序的数组&#xff0c;直到最…

【Android12】Android Framework系列---Adb和PMS安装apk源码流程

Adb和PMS安装apk源码流程 adb install命令 通过adb install命令可以将apk安装到Android系统&#xff08;注意&#xff1a;特定类型的apk&#xff0c;比如persist类型是无法通过adb安装的&#xff09; 下述命令中adb解析install命令&#xff0c;并调用Android PackageManagerS…

KAGGLE · GETTING STARTED CODE COMPETITION 图像风格迁移 示例代码阅读

本博文阅读的代码来自于I’m Something of a Painter Myself | Kaggle倾情推荐&#xff1a; Monet CycleGAN Tutorial | Kaggle 数据集说明 I’m Something of a Painter Myself | Kaggle Files monet_jpg - 300 Monet paintings sized 256x256 in JPEG formatmonet_tfrec -…

go语言(十一)----面向对象继承

一、面向对象继承 写一个父类 package mainimport "fmt"type Human struct {name stringsex string }func (this *Human) Eat() {fmt.Println("Human.Eat()...") }func (this *Human) Walk() {fmt.Println("Human.Walk()...") }func main() {h…

B(l)utter:一款针对Flutter移动端应用程序的逆向工程分析工具

关于B(l)utter B(l)utter是一款针对Flutter移动端应用程序的逆向工程分析工具&#xff0c;当前版本的B(l)utter仅支持Android libapp.so&#xff08;ARM64&#xff09;&#xff0c;可以帮助广大研究人员对基于Flutter开发的移动端应用程序进行逆向工程分析。 环境搭建 该应用…

dpdk网络转发环境的搭建

文章目录 前言ip命令的使用配置dpdk-basicfwd需要的网络结构测试dpdk-basicfwddpdk-basicfwd代码分析附录basicfwd在tcp转发时的失败抓包信息DPDK的相关设置 前言 上手dpdk有两难。其一为环境搭建。被绑定之后的网卡没有IP&#xff0c;我如何给它发送数据呢&#xff1f;当然&a…

全国各省市上市公司数量数据,Shp、excel格式,含上市企业数量、行政区划中心点位经纬度等字段

基本信息. 数据名称: 全国各省市上市公司数量数据 数据格式: Shp、excel 数据时间: 2023年1月 数据几何类型: 面 数据坐标系: WGS84 数据来源&#xff1a;网络公开数据 数据字段&#xff1a; 序号字段名称字段说明1province省份名称2provin_dm省份代码3city城市名…