C++模板类原理讲解
C++模板类原理讲解
C++模板是一种强大的编译期工具,它允许我们创建通用的、类型无关的类和函数。模板的主要目的是实现代码的重用和泛型编程。模板类的原理涉及以下几个方面:
- 模板的定义和实例化
- 模板的类型参数
- 模板特化
- 模板的编译过程
- 模板的优点和缺点
1. 模板的定义和实例化
模板是C++中用于创建泛型类和函数的机制。模板定义包含在关键字 template 后面的尖括号 < > 中,里面可以包含类型参数或非类型参数。
模板类定义示例:
template <typename T>
class GenericContainer {
private:T value;
public:GenericContainer(T value) : value(value) {}T getValue() const { return value; }void setValue(T value) { this->value = value; }void printValue() const { std::cout << "Value: " << value << std::endl; }
};
在上面的例子中,GenericContainer 是一个模板类,它可以接受任何类型 T。
模板实例化:
模板在使用时会被实例化。例如:
GenericContainer<int> intContainer(42);
GenericContainer<std::string> stringContainer("Hello, Templates!");
这里,GenericContainer<int> 和 GenericContainer<std::string> 分别实例化了 GenericContainer 模板类,生成了具体的类定义。
2. 模板的类型参数
模板参数可以是类型参数(如 typename T 或 class T),也可以是非类型参数(如 int N)。
类型参数示例:
template <typename T>
class GenericContainer {// ...
};
非类型参数示例:
template <typename T, int Size>
class Array {
private:T data[Size];
public:int getSize() const { return Size; }
};
3. 模板特化
模板特化是指为特定类型或值提供特定的实现。分为完全特化和部分特化。
完全特化示例:
template <>
class GenericContainer<std::string> {
private:std::string value;
public:GenericContainer(std::string value) : value(value) {}std::string getValue() const { return value; }void setValue(std::string value) { this->value = value; }void printValue() const { std::cout << "String Value: " << value << std::endl; }
};
4. 模板的编译过程
模板是在编译期处理的,这意味着模板代码在使用时会被实例化并生成具体的类或函数定义。这种编译期处理带来了灵活性和性能优势,但也增加了编译时间和代码膨胀的风险。
编译器在遇到模板定义时不会立即生成代码,而是在模板被实际使用(实例化)时生成具体的代码。这被称为“惰性实例化”。
5. 模板的优点和缺点
优点:
- 代码重用:模板允许编写一次代码,可以用于多种类型。
- 类型安全:模板在编译时进行类型检查,减少了运行时错误。
- 性能:模板在编译时实例化,生成的代码通常不会有运行时的开销。
缺点:
- 编译时间:模板的实例化会增加编译时间。
- 代码膨胀:大量的模板实例化可能导致二进制文件变大。
- 错误信息复杂:模板错误信息通常比较复杂,难以调试。
示例代码讲解
GenericContainer.h
#ifndef GENERICCONTAINER_H
#define GENERICCONTAINER_H#include <iostream>
#include <string>template <typename T>
class GenericContainer {
private:T value;
public:GenericContainer(T value) : value(value) {}T getValue() const { return value; }void setValue(T value) { this->value = value; }void printValue() const { std::cout << "Value: " << value << std::endl; }
};template <>
class GenericContainer<std::string> {
private:std::string value;
public:GenericContainer(std::string value) : value(value) {}std::string getValue() const { return value; }void setValue(std::string value) { this->value = value; }void printValue() const { std::cout << "String Value: " << value << std::endl; }
};#endif // GENERICCONTAINER_H
Collection.h
#ifndef COLLECTION_H
#define COLLECTION_H#include <vector>
#include <iostream>template <typename T>
class Collection {
private:std::vector<T> elements;
public:void addElement(T element) { elements.push_back(element); }void printElements() const {for (const auto& element : elements) {std::cout << element << std::endl;}}template <typename U>bool contains(const U& value) const {for (const auto& element : elements) {if (element == value) {return true;}}return false;}
};#endif // COLLECTION_H
main.cpp
#include "GenericContainer.h"
#include "Collection.h"int main() {GenericContainer<int> intContainer(42);intContainer.printValue();GenericContainer<std::string> stringContainer("Hello, Templates!");stringContainer.printValue();Collection<int> intCollection;intCollection.addElement(1);intCollection.addElement(2);intCollection.addElement(3);intCollection.printElements();std::cout << "Contains 2? " << intCollection.contains(2) << std::endl;std::cout << "Contains 4? " << intCollection.contains(4) << std::endl;Collection<std::string> stringCollection;stringCollection.addElement("Hello");stringCollection.addElement("World");stringCollection.printElements();std::cout << "Contains 'Hello'? " << stringCollection.contains(std::string("Hello")) << std::endl;std::cout << "Contains 'C++'? " << stringCollection.contains(std::string("C++")) << std::endl;return 0;
}
总结
C++模板类提供了一种强大的机制,用于编写类型无关和高度可复用的代码。理解模板的定义、实例化、特化以及编译过程,对于高效使用C++模板至关重要。虽然模板带来了许多优点,但也伴随着一些缺点,如编译时间增加和代码膨胀。通过合理的设计和使用,可以充分发挥模板的优势,减少其不足。
源代码
相关文章:
C++模板类原理讲解
C模板类原理讲解 C模板是一种强大的编译期工具,它允许我们创建通用的、类型无关的类和函数。模板的主要目的是实现代码的重用和泛型编程。模板类的原理涉及以下几个方面: 模板的定义和实例化模板的类型参数模板特化模板的编译过程模板的优点和缺点 1.…...
scratch编程03-反弹球
这篇文章和上一篇文章《scratch3编程02-使用克隆来编写小游戏》类似(已经完全掌握了克隆的可以忽略这篇文章),两篇文章都使用到了克隆来编写一个小游戏,这篇文章与上篇文章不同的是,本体在进行克隆操作时,不…...
postgresql数据库进阶知识
postgresql数据库进阶知识 # 如果表存在就先删除 drop table if exists student; # 创建学生表 # id serial not null 表示id自增 # id integer not null 表示id不自增 create table student (id serial not nullconstraint student_pkprimary…...
关于HTTP劫持,该如何理解、防范和应对
一、引言 HTTP劫持(HTTP Hijacking)是一种网络安全威胁,它发生在HTTP通信过程中,攻击者试图通过拦截、篡改或监控用户与服务器之间的数据流量,以达到窃取敏感信息或执行恶意操作的目的。今天我们就来详细了解HTTP劫持…...
System.Data.OracleClient.OracleException:“ORA-12571: TNS: 包写入程序失败
System.Data.OracleClient.OracleException:“ORA-12571: TNS: 包写入程序失败 解决方法: 首先%oracle_home%/network/admin下的sqlnet.ora文件,把SQLNET.AUTHENTICATION_SERVICES (NTS)加个 # 注释掉就好了...
saas产品运营案例 | 联盟营销计划如何帮助企业提高销售额?
在当今数字化时代,SaaS(软件即服务)产品已成为企业提高效率、降低成本的重要工具。然而,面对激烈的市场竞争,如何有效地推广SaaS产品、提高销售额,成为许多企业面临的挑战。林叔将以ClickFunnels为例&#…...
模式分解算法-满足3NF的无损且保持函数依赖的分解算法、满足BCNF的无损连接分解算法
一、引言 1、对指定的关系模式,若范式级别较低,为第一范式或第二范式,由于存在数据冗余或更新异常问题,在实际中一般是不可用的,关系模式的规范化就是将满足低一级的关系模式分解为若干满足高一级范式的关系模式的集合…...
荷兰与法国战平,双方能携手出现?
就在昨天晚上,荷兰队经历了90分钟的鏖战,最终0-0与法国队握手言和。此役,哈维-西蒙斯为荷兰队打进一球,但进球被判无效。从目前的积分形势来看,双方基本上确定携手晋级16强赛。本场比赛,荷兰队后卫内森-阿克…...
数据可视化实验二:回归分析、判别分析与聚类分析
目录 一、使用回归分析方法分析某病毒是否与温度呈线性关系 1.1 代码实现 1.2 线性回归结果 1.3 相关系数验证 二、使用判别分析方法预测某病毒在一定的温度下是否可以存活,分别使用三种判别方法,包括Fish判别、贝叶斯判别、LDA 2.1 数据集展示&am…...
FL论文专栏|设备异构、异步联邦
论文:Asynchronous Federated Optimization(12th Annual Workshop on Optimization for Machine Learning) 链接 实现Server的异步更新。每次Server广播全局Model的时候附带一个时间戳,Client跑完之后上传将时间戳和Model同时带回…...
【Java毕业设计】基于JavaWeb的礼服租赁系统
文章目录 摘 要Abstract目录1 绪论1.1 课题背景和意义1.2 国内外研究现状1.2.1 国外研究现状 1.3 课题主要内容 2 开发相关技术介绍2.1 Spring Boot框架2.2 Vue框架2.3 MySQL数据库2.4 Redis数据库 3 系统分析3.1 需求分析3.1.1 用户需求分析3.1.2 功能需求分析 3.2 可行性分析…...
代码随想录训练营Day 66|卡码网101.孤岛的总面积、102.沉没孤岛、103.水流问题、104.建造最大岛屿
1.孤岛的总面积 101. 孤岛的总面积 | 代码随想录 代码:(bfs广搜) #include <iostream> #include <vector> #include <queue> using namespace std; int dir[4][2] {1,0,0,1,-1,0,0,-1}; int count; void bfs(vector<vector<int>>&a…...
根据状态转移写状态机-二段式
目录 描述 输入描述: 输出描述: 描述 题目描述: 如图所示为两种状态机中的一种,请根据状态转移图写出代码,状态转移线上的0/0等表示的意思是过程中data/flag的值。 要求: 1、 必须使用对应类型的状…...
PyTorch C++扩展用于AMD GPU
PyTorch C Extension on AMD GPU — ROCm Blogs 本文演示了如何使用PyTorch C扩展,并通过示例讨论了它相对于常规PyTorch模块的优势。实验在AMD GPU和ROCm 5.7.0软件上进行。有关支持的GPU和操作系统的更多信息,请参阅系统要求(Linux…...
Hadoop archive
Index of /dist/hadoop/commonhttps://archive.apache.org/dist/hadoop/common/...
R语言——R语言基础
1、用repeat、for、while计算从1-10的所有整数的平方和 2、编写一个函数,给出两个正整数,计算他们的最小公倍数 3、编写一个函数,让用户输入姓名、年龄,得出他明年的年龄。用paste打印出来。例如:"Hi xiaoming …...
VFB电压反馈和CFB电流反馈运算放大器(运放)选择指南
VFB电压反馈和CFB电流反馈运算放大器(运放)选择指南 电流反馈和电压反馈具有不同的应用优势。在很多应用中,CFB和VFB的差异并不明显。当今的许多高速CFB和VFB放大器在性能上不相上下,但各有其优缺点。本指南将考察与这两种拓扑结构相关的重要考虑因素。…...
elasticsearch安装(centos7)
先给出网址 elasticsearch:Download Elasticsearch | Elastic elasticKibana:Download Kibana Free | Get Started Now | Elastic Logstash:Download Logstash Free | Get Started Now | Elastic ik分词:Releases infinilabs/…...
Java高手的30k之路|面试宝典|精通JVM(二)
JVM基本结构 类加载子系统:负责将.class文件加载到内存中,并进行验证、准备、解析和初始化。运行时数据区:包括堆(Heap)、方法区(Method Area)、Java栈(Java Stack)、本…...
JVM专题六:JVM的内存模型
前面我们通过Java是如何编译、JVM的类加载机制、JVM类加载器与双亲委派机制等内容了解到了如何从我们编写的一个.Java 文件最终加载到JVM里的,今天我们就来剖析一下这个Java的‘中介平台’JVM里面到底长成啥样。 JVM的内存区域划分 Java虚拟机(JVM&…...
国防科技大学计算机基础课程笔记02信息编码
1.机内码和国标码 国标码就是我们非常熟悉的这个GB2312,但是因为都是16进制,因此这个了16进制的数据既可以翻译成为这个机器码,也可以翻译成为这个国标码,所以这个时候很容易会出现这个歧义的情况; 因此,我们的这个国…...
微信小程序之bind和catch
这两个呢,都是绑定事件用的,具体使用有些小区别。 官方文档: 事件冒泡处理不同 bind:绑定的事件会向上冒泡,即触发当前组件的事件后,还会继续触发父组件的相同事件。例如,有一个子视图绑定了b…...
《Qt C++ 与 OpenCV:解锁视频播放程序设计的奥秘》
引言:探索视频播放程序设计之旅 在当今数字化时代,多媒体应用已渗透到我们生活的方方面面,从日常的视频娱乐到专业的视频监控、视频会议系统,视频播放程序作为多媒体应用的核心组成部分,扮演着至关重要的角色。无论是在个人电脑、移动设备还是智能电视等平台上,用户都期望…...
【Linux】C语言执行shell指令
在C语言中执行Shell指令 在C语言中,有几种方法可以执行Shell指令: 1. 使用system()函数 这是最简单的方法,包含在stdlib.h头文件中: #include <stdlib.h>int main() {system("ls -l"); // 执行ls -l命令retu…...
(二)TensorRT-LLM | 模型导出(v0.20.0rc3)
0. 概述 上一节 对安装和使用有个基本介绍。根据这个 issue 的描述,后续 TensorRT-LLM 团队可能更专注于更新和维护 pytorch backend。但 tensorrt backend 作为先前一直开发的工作,其中包含了大量可以学习的地方。本文主要看看它导出模型的部分&#x…...
Linux相关概念和易错知识点(42)(TCP的连接管理、可靠性、面临复杂网络的处理)
目录 1.TCP的连接管理机制(1)三次握手①握手过程②对握手过程的理解 (2)四次挥手(3)握手和挥手的触发(4)状态切换①挥手过程中状态的切换②握手过程中状态的切换 2.TCP的可靠性&…...
家政维修平台实战20:权限设计
目录 1 获取工人信息2 搭建工人入口3 权限判断总结 目前我们已经搭建好了基础的用户体系,主要是分成几个表,用户表我们是记录用户的基础信息,包括手机、昵称、头像。而工人和员工各有各的表。那么就有一个问题,不同的角色…...
Java - Mysql数据类型对应
Mysql数据类型java数据类型备注整型INT/INTEGERint / java.lang.Integer–BIGINTlong/java.lang.Long–––浮点型FLOATfloat/java.lang.FloatDOUBLEdouble/java.lang.Double–DECIMAL/NUMERICjava.math.BigDecimal字符串型CHARjava.lang.String固定长度字符串VARCHARjava.lang…...
高等数学(下)题型笔记(八)空间解析几何与向量代数
目录 0 前言 1 向量的点乘 1.1 基本公式 1.2 例题 2 向量的叉乘 2.1 基础知识 2.2 例题 3 空间平面方程 3.1 基础知识 3.2 例题 4 空间直线方程 4.1 基础知识 4.2 例题 5 旋转曲面及其方程 5.1 基础知识 5.2 例题 6 空间曲面的法线与切平面 6.1 基础知识 6.2…...
【AI学习】三、AI算法中的向量
在人工智能(AI)算法中,向量(Vector)是一种将现实世界中的数据(如图像、文本、音频等)转化为计算机可处理的数值型特征表示的工具。它是连接人类认知(如语义、视觉特征)与…...
