当前位置: 首页 > article >正文

【C++】模版(1)

目录

1. 泛型编程

2. 函数模版

2.1 函数模版概念

2.2 函数模版格式

2.3 函数模版的原理

2.4 函数模版实例化方式

隐式实例化

显式实例化

2.5 模版参数的匹配原则

3. 模版类

模版类的定义格式

模版类的实例化


1. 泛型编程

如何实现一个通用的交换函数呢?

void Swap(int& left, int& right)
{int temp = left;left = right;right = temp;
}
void Swap(double& left, double& right)
{double temp = left;left = right;right = temp;
}
void Swap(char& left, char& right)
{char temp = left;left = right;right = temp;
}

使用函数重载虽然可以实现,但是有一下几个不好的地方:

  • 重载的函数仅仅是类型不同,代码复用率比较低,只要有新类型出现时,就需要用户自己增加对应的函数。
  • 代码的可维护性比较低,一个出错可能所有的重载均出错。

泛型编程:编写与类型无关的通用代码,是代码复用的一种手段。模板是泛型编程的基础。

2. 函数模版

2.1 函数模版概念

函数模版是C++泛型编程的重要特性,它允许你编写一个通用的函数,这个函数可以处理不同的数据类型,而不需要为每种数据类型都单独编写一个函数。

2.2 函数模版格式

template< typename T1,typename T2,……,typename Tn>

返回值类型 函数名(参数列表){ }

//函数模版声明
template<typename T>
void Swap(T& x, T& y)
{T tmp = x;x = y;y = tmp;	
}

注意:typename是用来声明类型参数关键字,也可以使用class (template<class T>),切记不能使用struct代替 class。

2.3 函数模版的原理

函数模板是一个蓝图,它本身并不是函数,是编译器用使用方式产生特定具体类型函数的模具。 所以其实模板就是将本来应该我们做的重复的事情交给了编译器。

在编译阶段,当代码中调用函数模版时,编译器会根据传入的实参类型对函数模版进行实例化:

编译器会根据传入实参的类型,自动推导出模版中类型参数,template<typename T>里的T实际代表的数据类型,若调用Swap(d1,d2);编译器会将T推演为double。确定具体类型后,编译器会为该类型生成一份专门的函数代码,这个过程就是函数模版的实例化。

2.4 函数模版实例化方式

隐式实例化

让编译器根据实参推演模板参数的实际类型。


#include<iostream>
using namespace std;
//函数模版声明
template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a1 = 10, a2 = 20;double d1 = 10.2, d2 = 20.1;Add(a1, a2);//隐式实例化,编译器根据实参类型推导T为 intAdd(d1, d2);//隐式实例化,编译器根据实参类型推导T为 doublereturn 0;
}
#include<iostream>
using namespace std;
//函数模版声明
template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a1 = 10, a2 = 20;double d1 = 10.2, d2 = 20.1;//Add(a1, d1);  error!//当传入不同类型实参无法让编译器唯一确定模版参数类型时,编译报错,因为编译器一般不自动进行类型转换来确定函数模版 //可以用户自己强制转化Add(a1, (int)d1);return 0;
}

显式实例化

在函数名后的<>中指定模板参数的实际类型,告知编译器按指定类型对函数模版进行实例化。

#include<iostream>
using namespace std;
//函数模版声明
template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int main()
{int a1 = 2, a2 = 4;double d1 = 3.3, d2 = 5.5;cout << Add<int>(a1, d1) << endl;   //5   显式实例化,明确指定模版参数T为int  注意精度丢失问题cout << Add<double>(a2, d2) << endl;//9.5 显式实例化,明确指定模版参数T为doublereturn 0;
}
//函数模版声明
template<class T>
T*func(int n)
{return new T[n];
}int main()
{double* p1 = func<double>(10); //显示实例化return 0;
}

可在编译前期就确定函数模版实例化类型,有利于提高编译效率,减少编译时类型推导时间,显式实例化能让程序员精确控制模版参数类型。

2.5 模版参数的匹配原则

对于非模板函数和同名函数模板,如果其他条件都相同,在调动时会优先调用非模板函数而不会从该模板产生出一个实例。如果模板可以产生一个具有更好匹配的函数, 那么将选择模板。

#include<iostream>
using namespace std;
//函数模版声明
template<class T>
T Add(const T& left, const T& right)
{return left + right;
}int Add(const int& x, const int& y)
{return (x + y) * 5;
}int main()
{cout << Add(1, 2) << endl;    //15  与非函数模板类型完全匹配,不需要函数模板实例化cout << Add(2.2, 3.3) << endl;//5.5 模板函数可以生成更加匹配的版本,编译器根据实参生成更加匹配的Add函数return 0;
}

3. 类模版(模版类)

模版类的定义格式

template<class T1, class T2, ..., class Tn>
class 模板类名
{// 类内成员定义
};

模版类的实例化

模板类实例化与函数模板实例化不同,模板类实例化需要在模板类名字后跟<>,然后将实例化的类型放在<>中即可,模板类名字不是真正的类,而实例化的结果才是真正的类 (比如Satck是类名,而Satck<T>才是具体类型)。
模版类是模版,不是具体实例化的类。
代码示例: 
#include<iostream>
using namespace std;// 模版类
template<class T>
class Stack
{
public:Stack(int n = 4):_array(new T[n]), _size(0), _capacity(n){ }~Stack(){delete[]_array;_array = nullptr;_size = _capacity = 0;}void Push(const T& data);//类内声明private:T* _array;size_t _capacity;size_t _size;
};template<class T>    //在类外定义成员函数时,需要再次指定模版参数,因为函数定义是独立于类定义的作用域的,表明函数定义依赖模版参数。
void Stack<T>::Push(const T& data)//通过Stack<T>编译器知道这是模版类的成员函数,不是普通类的,如果省略<T>,编译器会认为Satck是一个已实例化的具体类
{                                 //通过T来确定具体是针对哪个类型参数实例化的类模版的成员函数进行定义if (_size == _capacity) //扩容{T* tmp = new T[_capacity * 2];memcpy(tmp, _array, sizeof(T) * _size);//将原数组的内容复制到新数组delete[]_array;//释放原数组_array = tmp;//更新指针和容量_capacity *= 2;}_array[_size++] = data;
} int main()
{//类模版通常需要显示实例化,根据实际使用的类型生成对应的代码Stack<int>st1;st1.Push(1);st1.Push(2);Stack<double>st2;st2.Push(1.1);st2.Push(2.2);return 0;
}
注意:模版不建议声明和定义分离到两个文件.h 和.cpp,会出现链接错误。

相关文章:

【C++】模版(1)

目录 1. 泛型编程 2. 函数模版 2.1 函数模版概念 2.2 函数模版格式 2.3 函数模版的原理 2.4 函数模版实例化方式 隐式实例化 显式实例化 2.5 模版参数的匹配原则 3. 模版类 模版类的定义格式 模版类的实例化 1. 泛型编程 如何实现一个通用的交换函数呢&#xff1f…...

基于开源AI智能名片链动2+1模式S2B2C商城小程序源码的去中心化商业扩散研究

摘要&#xff1a;本文探讨在去中心化商业趋势下&#xff0c;开源AI智能名片链动21模式S2B2C商城小程序源码如何助力企业挖掘数据价值、打破信息孤岛&#xff0c;实现商业高效扩散。通过分析该技术组合的架构与功能&#xff0c;结合实际案例&#xff0c;揭示其在用户关系拓展、流…...

5月19日day30打卡

模块和库的导入 知识点回顾&#xff1a; 导入官方库的三种手段导入自定义库/模块的方式导入库/模块的核心逻辑&#xff1a;找到根目录&#xff08;python解释器的目录和终端的目录不一致&#xff09; 作业&#xff1a;自己新建几个不同路径文件尝试下如何导入 一、导入官方库 …...

白杨SEO:不到7天,白杨SEO博客网站百度搜索显示和排名恢复正常!顺带说说上海线下GEO聚会分享和播客红利

大家好&#xff0c;我是白杨SEO&#xff0c;专注SEO十年以上&#xff0c;全网SEO流量实战派&#xff0c;AI搜索优化研究者。 5月开始&#xff0c;明显就忙起来了&#xff0c;不管是个人陪跑还是企业顾问&#xff0c;不管是需要传统SEO还是新媒体流量&#xff0c;还是当下这个A…...

Windows软件插件-音视频捕获

下载本插件 音视频捕获就是获取电脑外接的话筒&#xff0c;摄像头&#xff0c;或线路输入的音频和视频。 本插件捕获电脑外接的音频和视频。最多可以同时获取4个视频源和4个音频源。插件可以在win32和MFC程序中使用。 使用方法 首先&#xff0c;加载本“捕获”DLL&#xff0c…...

go 与面向对象编程(OOP)

Go 语言在设计上与传统面向对象&#xff08;OOP&#xff09;语言&#xff08;如 Java、C&#xff09;有明显差异&#xff0c;官方明确表示它并非纯面向对象语言。然而&#xff0c;它通过独特的方式实现了部分面向对象的核心特性。以下是关键分析&#xff1a; 1. Go 对传统 OOP…...

Mergekit——任务向量合并算法Ties解析

Mergekit——高频合并算法 TIES解析 Ties背景Ties 核心思想具体流程总结 mergekit项目地址 Mergekit提供模型合并方法可以概况为四大类&#xff1a;基本线性加权、基于球面插值、基于任务向量 以及一些专业化方法&#xff0c;今天我们来刷下基于任务向量的ties合并方法&#xf…...

Java 应用中的身份认证与授权:OAuth2.0 实现安全的身份管理

Java 应用中的身份认证与授权&#xff1a;OAuth2.0 实现安全的身份管理 在当今的软件开发领域&#xff0c;身份认证与授权是构建安全可靠应用的关键环节。而 Java 作为广泛使用的编程语言&#xff0c;在实现这一功能上有着诸多成熟的框架和方案。其中&#xff0c;OAuth2.0 凭借…...

【氮化镓】偏置对GaN HEMT 单粒子效应的影响

2025年5月19日,西安电子科技大学的Ling Lv等人在《IEEE Transactions on Electron Devices》期刊发表了题为《Single-Event Effects of AlGaN/GaN HEMTs Under Different Biases》的文章,基于实验和TCAD仿真模拟方法,研究了单粒子效应对关断状态、半开启状态和开启状态下AlG…...

Mysql 索引概述

索引&#xff08;index&#xff09;是帮助Mysql高效获取数据的数据结构 索引优点&#xff1a;1. 提高排序效率 2. 提高查询效率 索引缺点&#xff1a;1.索引占用空间&#xff08;可忽略&#xff09;2.索引降低了更新表的速度&#xff0c;如进行insert,update,delette 时效率降…...

HttpServletRequest常用功能简介-笔记

javax.servlet.http.HttpServletRequest 是 ServletRequest 接口的子接口&#xff0c;专用于处理 HTTP 协议相关的请求。它提供了访问请求行、请求头、请求参数以及请求属性等方法。 1.请求行&#xff08;Request Line&#xff09; ✅ 功能说明 请求行包含客户端发送的 HTTP …...

解决RAGFlow部署中镜像源拉取的问题

报错提示 Error response from daemon: Get "https://registry-1.docker.io/v2/ ": context deadline exceeded 解决方法 这个原因是因为拉取镜像源失败&#xff0c;可以在/etc/docker/daemon.json文件中添加镜像加速器&#xff0c;例如下面所示 {"registry…...

uniapp打包H5,输入网址空白情况

由于客户预算有限&#xff0c;最近写了两个uniapp打包成H5的案例&#xff0c;总结下面注意事项 1. 发行–网站-PCWeb或手机H5按钮&#xff0c;输入名称&#xff0c;网址 点击【发行】&#xff0c;生成文件 把这个给后端&#xff0c;就可以了 为什么空白呢 最重要一点&#xf…...

wsl2中Ubuntu22.04配置静态IP地址

第一步找到/etc/netplan目录下的01-netcfg.yaml文件&#xff0c;&#xff08;如果不存在则自己创建一个&#xff09;在里面配置一下代码&#xff1a; network:version: 2renderer: networkdethernets:eth0:dhcp4: noaddresses: [192.168.3.222/24]routes:- to: 0.0.0.0/0via: …...

C++(21):fstream的读取和写入

目录 1 ios::out 2 ios::in和is_open 3 put()方法 4 get()方法 4.1 读取单个字符 4.2 读取多个字符 4.3 设置终结符 5 getline() 1 ios::out 打开文件用于写入数据。如果文件不存在&#xff0c;则新建该文件&#xff1b;如果文件原来就存在&#xff0c;则打开时清除…...

NAT/代理服务器/内网穿透

目录 一 NAT技术 二 内网穿透/内网打洞 三 代理服务器 一 NAT技术 跨网络传输的时候&#xff0c;私网不能直接访问公网&#xff0c;就引入了NAT能讲私网转换为公网进行访问&#xff0c;主要解决IPv4(2^32)地址不足的问题。 1. NAT原理 当某个内网想访问公网&#xff0c;就必…...

Unity 多时间源Timer定时器实战分享:健壮性、高效性、多线程安全与稳定性能全面解析

简介 Timer 是一个 Unity 环境下高效、灵活的定时任务调度系统&#xff0c;支持以下功能&#xff1a; •支持多种时间源&#xff08;游戏时间 / 非缩放时间 / 真实时间&#xff09; •支持一次性延迟执行和重复执行 •提供 ID、回调、目标对象等多种查询和销毁方式 •内建…...

深入解析Spring Boot与Spring Security的集成实践

深入解析Spring Boot与Spring Security的集成实践 引言 Spring Security是Spring生态中用于处理认证和授权的强大框架。在Spring Boot项目中集成Spring Security可以轻松实现用户认证、权限控制等功能。本文将详细介绍如何从零开始集成Spring Security&#xff0c;并解决实际…...

【iOS】探索消息流程

探索消息流程 Runtime介绍OC三大核心动态特性动态类型动态绑定动态语言 方法的本质代码转换objc_msgSendSELIMPMethod 父类方法在子类中的实现 消息查找流程开始查找快速查找流程慢速查找流程二分查找方法列表父类缓存查找 动态方法解析动态方法决议实例方法类方法优化 消息转发…...

用户账号及权限管理:企业安全的基石与艺术

在当今数字化时代,用户账号及权限管理已成为企业IT安全体系中不可或缺的核心组件。它不仅是保护敏感数据的第一道防线,更是确保业务运营效率和合规性的关键。本文将深入探讨用户账号及权限管理的重要性、最佳实践以及实施策略,助您构建一个安全、高效且灵活的访问控制体系。…...

413 Payload Too Large 问题定位

源头 一般是服务器或者nginx 配置导致的 nginx http {client_max_body_size 50m; # 调整为所需大小&#xff08;如 50MB&#xff09;# 其他配置... }nginx 不配置&#xff0c;默认是1M 服务器 spring 不配置也是有默认值的好像也是1M 如果出现413 可以试着修改配置来避…...

2025年渗透测试面试题总结-360[实习]安全工程师(题目+回答)

网络安全领域各种资源&#xff0c;学习文档&#xff0c;以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各种好玩的项目及好用的工具&#xff0c;欢迎关注。 目录 1. 自我介绍 2. WAF及其绕过方式 3. IPS/IDS/HIDS 4. 云安全 5. 绕过安骑士/安全狗 6. Gopher扩展…...

Ubuntu16.04升级gcc/g++版本方法

0 前言 gcc与g分别是GNU的c和c编译器&#xff0c;Ubuntu16.04默认的gcc和g的版本是5.4.0&#xff0c;在使用一些交叉编译工具链会提示找不到GLIBC_2.27&#xff0c;而GLIBC_2.27又需要gcc 6.2以上版本&#xff0c;因此本文介绍Ubuntu16.04升级gcc/g版本的方法。 1 Ubuntu16.0…...

微信小程序van-dialog确认验证失败时阻止对话框的关闭

使用官方(Vant Weapp - 轻量、可靠的小程序 UI 组件库)的before-close&#xff1a; wxml&#xff1a; <van-dialog use-slot title"名称" show"{{ show }}" show-cancel-button bind:cancel"onClose" bind:confirm"getBackInfo"…...

边缘计算模块

本文来源 &#xff1a;腾讯元宝 边缘计算模块是一种部署在网络边缘&#xff08;靠近数据源&#xff09;的集成化硬件/软件设备&#xff0c;用于实时处理本地数据&#xff0c;减少云端依赖&#xff0c;提升响应速度与安全性。以下是其核心要点&#xff1a; ​​1. 核心组成​​ …...

【极兔快递Java社招】一面复盘|数据库+线程池+AQS+中间件面面俱到

【极兔快递Java社招】一面复盘&#xff5c;数据库线程池AQS中间件面面俱到 &#x1f4cd;面试公司&#xff1a;极兔快递 &#x1f45c;面试岗位&#xff1a;Java后端开发工程师 &#x1f550;面试时长&#xff1a;约 60 分钟 &#x1f504;面试轮次&#xff1a;第 1 轮技术面&…...

OceanBase 的系统变量、配置项和用户变量有何差异

在继续阅读本文之前&#xff0c;大家不妨先思考一下&#xff0c;数据库中“系统变量”、“用户变量”以及“配置项”这三者之间有何不同。如果感到有些模糊&#xff0c;那么本文将是您理清这些概念的好帮手。 很多用户在使用OceanBase数据库中的“配置项”和“系统变量”&#…...

Git本地使用小Tips

要将本地仓库 d:\test 的更新推送到另一个本地仓库 e:\test&#xff0c;可以使用 Git 的远程仓库功能。以下是具体步骤&#xff1a; ​​在 e:\test 中添加 d:\test 作为远程仓库​​ 在 e:\test 目录中打开 Git Bash 或命令行&#xff0c;执行以下命令&#xff1a; git remo…...

【Python】Jupyter指定具体路径

一、右键Jupyter Notebook 右击Jupyter Notebook点击属性 二、修改以下两个地方...

ThreadLocal作一个缓存工具类

1、工具类 import java.util.HashMap; import java.util.Map;public class ThreadLocalUtil {// 使用Map存储多类型数据private static final ThreadLocal<Map<String, Object>> CONTEXT_HOLDER new ThreadLocal<>();// 存储数据public static void set(Str…...