C++面向对象(OOP)编程-模板

news2025/1/22 17:04:40

本文主要讲解C++的模板,其中包括模板的分类,函数模板和类模板,以及类模板与友元函数关系引起的几种关系。强调提供代码来搞懂C++模板这一泛型编程手段。

目录

1 C++模板

2 模板的本质

3 模板分类

4 函数模板

4.1 函数模板定义格式

4.2 函数模板的实例化

4.3 函数模板和一般函数的对比

4.4 函数模板使用规则

5 类模板

5.1 类模板定义格式

5.2 类模板的实例化

5.3 类模板使用规则

5.4 类模板与友元函数

5.4.1 非模板友元函数

5.4.2 约束模板友元函数

5.4.3 非约束模板友元函数

5.4.4 三种类模板与友元函数


1 C++模板

        C++模板是一种泛型编程技术,它允许程序员编写通用的代码,可以处理不同类型的数据。模板的主要目的是实现代码重用和类型安全。

        函数、类以及类继承为程序的代码复用提供了基本手段,还有一种代码复用途径——类属类型(泛型),类型作为参数实现代码的复用。

2 模板的本质

        模板是为实现代码复用的一种手段,程序处理的数据有两个基本的特征:值和类型。常规的函数等将值作为参数来进行传递,实现了不同值的改变。由此引入的问题是:是否可以将类型作为参数来传入?这是模板解决的问题。因此,模板的本质是数据类型的参数化。

        需要注意的是:一个模板并非一个实实在在的类或函数,仅仅是一个类或函数的描述。

3 模板分类

         模板分为函数模板和类模板。

         (1)函数模板:函数模板是一个通用的函数,它可以处理不同类型的参数。要定义一个函数模板,需要在函数名后面加上尖括号<>,并在其中放置模板参数。

        简单的函数模板的例子:

template <typename T>
void swap(T &a, T &b) {
    T temp = a;
    a = b;
    b = temp;
}

        上述例子定义一个模板函数,可以交换不同类型的两个数据。 

        (2)类模板:类模板是一个通用的类,它可以处理不同类型的数据成员和成员函数。要定义一个类模板,需要在类名后面加上尖括号<>,并在其中放置模板参数。

        一个简答的类模板的例子:

template <typename T>
class Stack {
public:
    void push(const T &value);
    T pop();
    T top() const;
    bool empty() const;
private:
    std::vector<T> data;
};

        上述例子定义一个Stack类,类中成员和函数的变量的类型是可变的。

4 函数模板

4.1 函数模板定义格式

template <typename 形参名, typename 形参名...>     //模板头(模板说明)

返回值类型  函数名(参数列表)                   //函数定义

{

        函数体;

}

        (1) template是声明模板的关键字,告诉编译器开始泛型编程。

        (2)尖括号<>中的typename是定义形参的关键字,用来说明其后的形参名为类型 参数,(模板形参)。Typename(建议用)可以用class关键字代替,两者没有区别。

        (3)模板形参(类属参数)不能为空(俗成约定用一个大写英文字母表示),且在函 数定义部分的参数列表中至少出现一次。与函数形参类似,可以用在函数定义的各 个位置:返回值、形参列表和函数体。

        (4)函数定义部分:与普通函数定义方式相同,只是参数列表中的数据类型要使用 尖括,号<>中的模板形参名来说明。当然也可以使用一般的类型参数。

template <typename T>
T add(T a, T b)
{
    return (a+b);
}

        函数模板的调用使用实参来推演进行

        cout << "a+b: " << add(15,34) << endl;  // 这是行的,都按照int类型输入

        cout << "a+b: " << add(12.3,34.0) << endl;  // 这是可行的,都按照double或者float类型输入

        cout << "a+b: " << add(12,34.4) << endl;//这是不可行的,两个实参的类型不一样,无法判断

4.2 函数模板的实例化

        (1)隐式实例化

                形如  add(15,34) 这种就是隐式实例化,根据输入的15和34来推断类型,将模板实例化成一个int类型的函数,然后将实参传入,不能为同一个模板指定两种类型。

        代码如下:

#include <iostream>
using namespace std;

/*
    C++模板:
        (1) 函数模板
        (2) 类模板
*/

template <typename T>
T add(T a, T b)
{
    return (a+b);
}

int main (int argc, char *argv[])
{
    cout << "a+b: " << add(12.3,34.0) << endl;
    return 0;
}

        运行结果:

        (2)显示实例化

                形如add<float>(12,34.4) 会指定模板的参数的类型为float,将模板实例化成一个float类型的函数,然后将12 和34.4以float类型输入,可以为同一个模板指定两种不同的类型。

        代码如下:

#include <iostream>
using namespace std;

/*
    C++模板:
        (1) 函数模板
        (2) 类模板
*/

template <typename T>
T add(T a, T b)
{
    return (a+b);
}

int main (int argc, char *argv[])
{
    cout << "a+b: " << add<float>(12,34.4) << endl;
    return 0;
}

        运行结果:

        注意:函数模板仍然支持函数重载,仅仅只是数据类型不确定。

4.3 函数模板和一般函数的对比

        两者的区别:函数模板不允许自动类型转化,普通函数则能进行自动类型转换

        函数模板和普通函数在一起时的调用规则:

        函数模板可以像普通函数一样被重载

        C++编译器优先考虑普通函数

        如果函数模板可以产生一个更好的匹配,那么选择模板

        可以通过空模板实参列表的语法限定编译器只通过模板匹配

4.4 函数模板使用规则

        (1)函数模板中的每一个类型参数在函数参数表中必须至少使用一次。

        (2)在全局域中声明的与模板参数同名的对象、函数或类型,在函数模板中将被 隐藏。即同名的变量在模板中,优先确定为适配模板类型的变量与全局变量没有任何关系。

        (3)函数模板中定义声明的对象或类型不能与模板参数同名

        (4)模板参数名在同一模板参数表中只能使用一次,但可在多个函数模板声明或 定义之间重复使用如下:

template <typename T, typename T>  //错误,在同一个模板中重复定义模板参数

void fun1(T t1, T t2) { }

template<typename T>

void fun2(T t1) { }

template <typename T>    //在不同函数模板中可重复使用相同模板参数名

void fun3(T t3) { }

        (5)模板的定义和多处声明所使用的模板参数名不一定要必须相同。如下:

//模板的前向声明

template <typename T>

void func1(T t1,T t2,T t3);

//模板的定义

template <typename U>

void func1(U t1, U t2, U t3)

{

    //……

}

        (6)函数模板如果有多个模板参数,则每个模板类型前都必须使用关键字typename 或class修饰。例如:

template <typename T, class U>   //两个关键字可以混用

void func(T t, U u)  { }

template <typename T,U>        //错误,每一个模板参数前都要有关键字修饰

void func(T t, U u)  { }

5 类模板

5.1 类模板定义格式

template<typename 形参名,typename 形参名…>

class 类名

{

        ………

}

        (1)template类模板中的关键字含义与函数模板相同。

        (2)类模板中的类型参数可用在类声明和类实现中。

        (3)类模板的模板形参(类型参 数)不能为空,一旦声明了类模板就可以用类模板的形参名声明类中的成员变量和 成员函数,即在类中使用内置数据类型的地方都可以使用模板形参名来代替。

5.2 类模板的实例化

         类模板包含参数,是类的抽象,类模板是类的抽象,类是对象的抽象。

        类模板的实例化必须指明数据类型,不支持隐式实例化。

        一个例子:

#include <iostream>
#include <string>
using namespace std;

template < class T,class U>
class A
{
    private:
        T t1;
        T t2;
        U name;
    public:
        A(T a, T b): t1(a), t2(b) {cout << "有参构造函数" << endl;}
        T add()
        {
            return (t1+t2);
        }
        T sub();

        void print(U str)
        {
            cout << str << endl;
        }

};

// 类模板成员函数外部定义
template < class T,class U>
T A<T,U>::sub()
{
    return (A<T,U>::t1 - A<T,U>::t2);
}


int main(int argc, char *argv[])
{

    {
        A<int,string> a1(23,45);
        A<int,string> a2(23,45.9);
        cout << "a1->a+b: " << a1.add() << endl;
        cout << "a1->a-b: " << a1.sub() << endl;
        cout << "a2->a+b: " << a2.add() << endl;

        a1.print("Hello Gui Yang Yang!");
         
    }

    return 0;
}

        由上述代码可以看到类模板的实例化必须要指明类型参数,并且由于类模板会指定类中成员变量和成员函数的类型,因此,对于调用其中的成员函数时,已经被指定了类型,是显示的调用,不存在类型的推导。

        对于类模板外部定义的成员函数,注意要加上类的参数类型。如上述代码中sub()函数

        运行结果:

5.3 类模板使用规则

        模板的声明或定义只能在全局、命名空间或类范围内进行,不能在局部范围、函数 内进行,不能在普通函数或者main函数中声明。声明或定义一个模板还有 以下几点需注意:

        (1)如果在全局域中声明了与模板参数同名的变量,则该变量被隐藏,优先考虑模板内部。

        (2)模板参数名不能被当作类模板定义中类成员的名字。

        (3)同一个模板参数名在模板参数表中只能出现一次。

        (4)在不同的类模板声明或定义中,模板参数名可以被重复使用。

5.4 类模板与友元函数

5.4.1 非模板友元函数

        非模板友元就是在类模板中声明普通的友元函数和友元类。声明的友元函数是一个普通的函数,类型本身不是模板,但是可以传递模板类的参数。

        一个例子:

#include <iostream>
using namespace std;
template<typename T>
class Person
{
	T x;
public:
	Person(const T& t)
	{
		x = t;
	}
	friend void log(const Person<T>& a);//有参友元函数
};
void log(const Person<int>& a)//模板形参为int类型(显示具体化)
{
	cout << "int:" << a.x << endl;
}
void log(const Person<double>& a)//模板形参为double类型(显示具体化)
{
	cout << "double:" << a.x << endl;
}
int main()
{
	Person<int> a(10);//创建int类型对象
	log(a);
	Person<double> b(12.5);//创建double类型对象
	log(b);
	return 0;
}

        运行结果:

5.4.2 约束模板友元函数

        这种友元函数本身就是一个函数模板,但其实例化类型取决于类被实例化时的类型 (被约束)。每个类的实例化都会产生一个与之匹配的具体化的友元函数。关键是友元函数本身是模板,类型是与类的类型一致。

一个例子:

#include <iostream>
using namespace std;
//(1)函数模板声明
template<typename T>
void func();
template<typename T>
void show(T& t);
//类模板定义
template <typename U>
class A
{
private:
	U item;
	static int count;
public:
	A(const U& u) :item(u) { count++; }
	~A() { count--; }
	//(2)在类模板中将函数模板声明为类的友元函数
	friend void func<U>();   //友元函数模板
	friend void show<>(A<U>& a);   //友元函数模板
};
template<typename T>
int A<T>::count = 0;     //类A的T类型对象的个数
//(3)友元函数模板的定义
template<typename T>
void func()
{
	cout << "template size: " << sizeof(A<T>) << ";";
	cout << " template func(): " << A<T>::count << endl;
}
template<typename T>
void show(T& t)
{
	cout << t.item << endl;
}
int main()
{
	func<int>();    //调用int类型的函数模板实例,int类型,其大小为	4字节
	A<int> a(10);   //定义类对象
	A<int> b(20);
	A<double> c(1.2);
	show(a);   //调用show()函数,输出类对象的数据成员值
	show(b);
	show(c);
	cout << "func<int> output:\n";
	func<int>();    //运行到此,已经创建了两个int类型对象
	cout << "func<double>() output:\n";
	func<double>();
	return 0;
}

        运行结果:

        上述代码将func()与show()函数定义成了模板并声明为类的友元,在定义函数模 板时是在类外定义的,当调用函数时,func()函数后带有<>说明函数的实例化 类型,而show()是直接调用的。

5.4.3 非约束模板友元函数

        在类内部声明友元函数模板,友元函数的模板形参与类模板的形参没有联系,此时 友元函数为类模板的非约束模板友元函数。就是单独给友元函数定义了新的模板与类的模板的参数没有任何的关系。

一个例子:

#include <iostream>
using namespace std;
template<typename T>
class Person
{
private:
	T item;
public:
	Person(const T& t) :item(t) {}
	template<typename U, typename V>  //在类内部声明函数模板
	friend void show(U& u, V& v);
};
template<typename U, typename V>
void show(U& u, V& v)
{
	cout << u.item << "," << v.item << endl;
}
int main()
{
	Person<int> a(10);
	Person<int> b(20);
	Person<double> c(1.2);
	cout << "a,b: ";
	show(a, b);
	cout << "a,c: ";
	show(a, c);
	return 0;
}

        运行结果:

        上述代码分析:函数模板的形参类型与类模板的形参类型不相关,因此,它可以接受任何类型的参数:第一次调用传入的是两个int类型的类对象,第二次调用传入的是一个 int类型和一个double类型的对象。

5.4.4 三种类模板与友元函数

        这三种的区别,关键是其友元函数的参数是否与类模板的参数有关,并且友元函数是否为模板函数。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Android 一分钟使用RecyclerView完美实现瀑布

【免费】安卓RecyclerView瀑布流效果实现资源-CSDN文库 1.WaterfallFlowActivity 主函数代码&#xff1a; package com.example.mytestapplication;import android.os.Bundle; import android.util.Log; import android.view.LayoutInflater; import android.widget.Toast;im…

vs code调试.so文件

使用vs code调试.so文件 1 vs code中安装c的debug插件2 【重要】编写launch.json3 在.so的源码中打断点4 debug模式启动进程5 attach进程6 开始调试 .so是一种动态链接库&#xff0c;在大型项目以及跨语言项目中经常用到。在拿到.so文件对应的源码后进行debug呢&#xff1f; 简…

天猫数据分析平台-天猫销售数据查询软件-11月天猫平台冲锋衣市场销售运营数据分析

随着气温逐渐下降&#xff0c;保暖服饰迎来热销&#xff0c;冲锋衣的需求大增。如今冲锋衣已经不仅仅是户外运动的装备&#xff0c;还成为很多年轻人的日常穿搭和时尚的追求。 新的穿搭趋势也带来了巨大的市场机会。据公开数据显示&#xff0c;中国有冲锋衣生产及经营企业超过8…

SpringBoot之响应案例的详细解析

2.3 案例 下面我们通过一个案例&#xff0c;来加强对请求响应的学习。 2.3.1 需求说明 需求&#xff1a;加载并解析xml文件中的数据&#xff0c;完成数据处理&#xff0c;并在页面展示 获取员工数据&#xff0c;返回统一响应结果&#xff0c;在页面渲染展示 2.3.2 准备工作…

Missing artifact org.wltea.analyzer:ik-analyzer:jar:5.0

没有找到【org.wltea.analyzer】 找到了【org.wltea.ik-analyzer】 https://github.com/wks/ik-analyzer https://github.com/wks/ik-analyzer.git https://code.google.com/archive/p/ik-analyzer/downloads?page2 C:\Users\Administrator\Desktop\ik-analyzer-master>m…

LabVIEW软件模拟氢燃料电池在车辆中的应用

LabVIEW软件模拟氢燃料电池在车辆中的应用 在追求可持续能源的时代&#xff0c;氢燃料电池在绿色经济中扮演着关键角色。本研究通过LabVIEW软件模拟和评估了氢燃料电池在车辆应用中的性能和效率。LabVIEW作为一个强大的模拟工具&#xff0c;能够动态模拟氢燃料电池系统在不同条…

Hugging Face实战-系列教程19:文本摘要建模实战1 之 数据清洗(中文商城评价数据处理方法)

&#x1f6a9;&#x1f6a9;&#x1f6a9;Hugging Face 实战系列 总目录 有任何问题欢迎在下面留言 本篇文章的代码运行界面均在Jupyter Notebook中进行 本篇文章配套的代码资源已经上传 文本摘要建模实战1 之 数据清洗 文本摘要建模实战2 之 Tokenizer处理 1 任务概述 1.1 任…

『踩坑记录』IDEA Spring initialzr新建Spring项目不能选择jdk8的解决方法

问题描述 Spring initializr新建Spring项目不能选低版本java 解决方法 默认官方start.spring.io已不支持自动生成低版本jkd的Spring项目&#xff0c;自定义用阿里云的starter即可 用阿里云的就能支持低版本jdk了 完 欢迎关注我的CSDN博客 &#xff1a;Ho1aAs 版权属于&a…

【C++】POCO学习总结(十九):哈希、URL、UUID、配置文件、日志配置、动态库加载

【C】郭老二博文之&#xff1a;C目录 1、哈希 1.1 说明 std::map和std::set 的性能是&#xff1a;O(log n) POCO哈希的性能比STL容器更好&#xff0c;大约快两&#xff1b; POCO中对应std::map的是&#xff1a;Poco::HashMap&#xff1b; POCO中对应std::set的是 Poco::Hash…

【04】GeoScene导出海图或者电子航道图000数据成果

1创建一个带有覆盖面和定义的产品 如果你没有已存在的S-57数据&#xff0c;你可以通过捕捉新的产品覆盖范围&#xff08;多边形产品范围&#xff09;及其所需的产品定义信息&#xff08;产品元数据&#xff09;来为新产品创建基础。 注&#xff1a; 如果你已经有一个S-57数据…

3800个字彻底弄清cortex

3800个字彻底弄清cortex arm内核发展历史cortexM0系列芯片系统框图通用寄存器m0特殊寄存器m3/m4/m7特殊寄存器 MSP和PSPxPSRPRIMASKCONTROLFAULTMASKBASEPRI 栈空间操作异常和中断 系统异常 NVIC可嵌套向量中断控制器系统操作寄存器 NVIC寄存器系统控制块SCB寄存器SysTick寄存…

算法训练第四十一天|343. 整数拆分、96. 不同的二叉搜索树

343. 整数拆分&#xff1a; 题目链接 给定一个正整数 n &#xff0c;将其拆分为 k 个 正整数 的和&#xff08; k > 2 &#xff09;&#xff0c;并使这些整数的乘积最大化。 返回 你可以获得的最大乘积 。 示例 : 输入: n 2 输出: 1 解释: 2 1 1, 1 1 1。解答&…

银行测试:第三方支付平台业务流,功能/性能/安全测试方法(超详细整理)

1、第三方支付平台的功能和结构特点 在信用方面&#xff0c;第三方支付平台作为中介&#xff0c;在网上交易的商家和消费者之间作一个信用的中转&#xff0c;通过改造支付流程来约束双方的行为&#xff0c;从而在一定程度上缓解彼此对双方信用的猜疑&#xff0c;增加对网上购物…

IDEA报错处理

问题1 IDEA 新建 Maven 项目没有文件结构 pom 文件为空 将JDK换成1.8后解决。 网络说法&#xff1a;别用 java18&#xff0c;换成 java17 或者 java1.8 都可以&#xff0c;因为 java18 不是 LTS 版本&#xff0c;有着各种各样的问题。。

PowerShell实战:Get-Content命令使用详解

目录 一、Get-Content介绍 二、语法格式 三、参数详解 四、使用案例 4.1 获取文件内容 4.2 获取文件前三行内容 4.3 获取文件最后三行内容 4.4通过管道方式获取最后两行内容 4.5使用逗号作为分隔符 4.6 Filter方式读取多个文件 4.7 Include方式读取多个文件 一、Get-Content介绍…

安装android studio

记录一下安装android studio的过程&#xff1a; 1.首先安装android studio到某一文件夹后&#xff0c;在C盘用户目录下可以看到.android文件夹。C:\Users\22515\AppData\Local\Google目录下也会出现AndroidStudio2022.2文件夹。&#xff08;注意&#xff1a;用户名&#xff0c…

还在为学MyBatis发愁?史上最全,一篇文章带你学习MyBatis

文章目录 前言一、&#x1f4d6;MyBatis简介1.Mybatis历史2.MyBatis特性3.对比&#xff08;其他持久化层技术&#xff09; 二、&#x1f4e3;搭建MyBatis1.开发环境2.创建maven工程3.创建MyBatis核心配置文件4.创建mapper接口5.创建MyBatis的映射文件6.通过junit测试功能7.加入…

lambda自定义比较规则-sort函数或优先队列

Lambda表达式的一般形式为 [captures](params){body}对于优先队列的自定义排序规则&#xff0c;常见方法是写成结构体形式 struct cmp{bool operator()(pair<int,int> map1,pair<int,int> map2){return map1.second>map2.second;} }; priority_queue<pair&…

【C语言】自定义类型——枚举、联合体

引言 对枚举、联合体进行介绍&#xff0c;包括枚举的声明、枚举的优点&#xff0c;联合体的声明、联合体的大小。 ✨ 猪巴戒&#xff1a;个人主页✨ 所属专栏&#xff1a;《C语言进阶》 &#x1f388;跟着猪巴戒&#xff0c;一起学习C语言&#x1f388; 目录 引言 枚举 枚举…

利用原始套接字解决mac地址错误问题【南瑞SysKeeper-2000】

一&#xff1a;案例描述 一键可视顺控图像智能项目在网络部署过程中&#xff0c;对网络限制隔离安全性要求很高&#xff0c;用到正向隔离装置&#xff08;南瑞SysKeeper-2000型号&#xff09;。 图一 正向装置示意图 现场发现问题&#xff1a;直连网线情况下&#xff0c;我方…