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&…...

从深圳崛起的“机器之眼”:赴港乐动机器人的万亿赛道赶考路
进入2025年以来,尽管围绕人形机器人、具身智能等机器人赛道的质疑声不断,但全球市场热度依然高涨,入局者持续增加。 以国内市场为例,天眼查专业版数据显示,截至5月底,我国现存在业、存续状态的机器人相关企…...

LeetCode - 394. 字符串解码
题目 394. 字符串解码 - 力扣(LeetCode) 思路 使用两个栈:一个存储重复次数,一个存储字符串 遍历输入字符串: 数字处理:遇到数字时,累积计算重复次数左括号处理:保存当前状态&a…...
鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个医院查看报告小程序
一、开发环境准备 工具安装: 下载安装DevEco Studio 4.0(支持HarmonyOS 5)配置HarmonyOS SDK 5.0确保Node.js版本≥14 项目初始化: ohpm init harmony/hospital-report-app 二、核心功能模块实现 1. 报告列表…...
今日学习:Spring线程池|并发修改异常|链路丢失|登录续期|VIP过期策略|数值类缓存
文章目录 优雅版线程池ThreadPoolTaskExecutor和ThreadPoolTaskExecutor的装饰器并发修改异常并发修改异常简介实现机制设计原因及意义 使用线程池造成的链路丢失问题线程池导致的链路丢失问题发生原因 常见解决方法更好的解决方法设计精妙之处 登录续期登录续期常见实现方式特…...
Java毕业设计:WML信息查询与后端信息发布系统开发
JAVAWML信息查询与后端信息发布系统实现 一、系统概述 本系统基于Java和WML(无线标记语言)技术开发,实现了移动设备上的信息查询与后端信息发布功能。系统采用B/S架构,服务器端使用Java Servlet处理请求,数据库采用MySQL存储信息࿰…...
【无标题】路径问题的革命性重构:基于二维拓扑收缩色动力学模型的零点隧穿理论
路径问题的革命性重构:基于二维拓扑收缩色动力学模型的零点隧穿理论 一、传统路径模型的根本缺陷 在经典正方形路径问题中(图1): mermaid graph LR A((A)) --- B((B)) B --- C((C)) C --- D((D)) D --- A A -.- C[无直接路径] B -…...

LLMs 系列实操科普(1)
写在前面: 本期内容我们继续 Andrej Karpathy 的《How I use LLMs》讲座内容,原视频时长 ~130 分钟,以实操演示主流的一些 LLMs 的使用,由于涉及到实操,实际上并不适合以文字整理,但还是决定尽量整理一份笔…...

从 GreenPlum 到镜舟数据库:杭银消费金融湖仓一体转型实践
作者:吴岐诗,杭银消费金融大数据应用开发工程师 本文整理自杭银消费金融大数据应用开发工程师在StarRocks Summit Asia 2024的分享 引言:融合数据湖与数仓的创新之路 在数字金融时代,数据已成为金融机构的核心竞争力。杭银消费金…...
PostgreSQL——环境搭建
一、Linux # 安装 PostgreSQL 15 仓库 sudo dnf install -y https://download.postgresql.org/pub/repos/yum/reporpms/EL-$(rpm -E %{rhel})-x86_64/pgdg-redhat-repo-latest.noarch.rpm# 安装之前先确认是否已经存在PostgreSQL rpm -qa | grep postgres# 如果存在࿰…...

手机平板能效生态设计指令EU 2023/1670标准解读
手机平板能效生态设计指令EU 2023/1670标准解读 以下是针对欧盟《手机和平板电脑生态设计法规》(EU) 2023/1670 的核心解读,综合法规核心要求、最新修正及企业合规要点: 一、法规背景与目标 生效与强制时间 发布于2023年8月31日(OJ公报&…...