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&…...
Golang 面试经典题:map 的 key 可以是什么类型?哪些不可以?
Golang 面试经典题:map 的 key 可以是什么类型?哪些不可以? 在 Golang 的面试中,map 类型的使用是一个常见的考点,其中对 key 类型的合法性 是一道常被提及的基础却很容易被忽视的问题。本文将带你深入理解 Golang 中…...
PHP和Node.js哪个更爽?
先说结论,rust完胜。 php:laravel,swoole,webman,最开始在苏宁的时候写了几年php,当时觉得php真的是世界上最好的语言,因为当初活在舒适圈里,不愿意跳出来,就好比当初活在…...
【SQL学习笔记1】增删改查+多表连接全解析(内附SQL免费在线练习工具)
可以使用Sqliteviz这个网站免费编写sql语句,它能够让用户直接在浏览器内练习SQL的语法,不需要安装任何软件。 链接如下: sqliteviz 注意: 在转写SQL语法时,关键字之间有一个特定的顺序,这个顺序会影响到…...
IT供电系统绝缘监测及故障定位解决方案
随着新能源的快速发展,光伏电站、储能系统及充电设备已广泛应用于现代能源网络。在光伏领域,IT供电系统凭借其持续供电性好、安全性高等优势成为光伏首选,但在长期运行中,例如老化、潮湿、隐裂、机械损伤等问题会影响光伏板绝缘层…...
均衡后的SNRSINR
本文主要摘自参考文献中的前两篇,相关文献中经常会出现MIMO检测后的SINR不过一直没有找到相关数学推到过程,其中文献[1]中给出了相关原理在此仅做记录。 1. 系统模型 复信道模型 n t n_t nt 根发送天线, n r n_r nr 根接收天线的 MIMO 系…...
初探Service服务发现机制
1.Service简介 Service是将运行在一组Pod上的应用程序发布为网络服务的抽象方法。 主要功能:服务发现和负载均衡。 Service类型的包括ClusterIP类型、NodePort类型、LoadBalancer类型、ExternalName类型 2.Endpoints简介 Endpoints是一种Kubernetes资源…...
TSN交换机正在重构工业网络,PROFINET和EtherCAT会被取代吗?
在工业自动化持续演进的今天,通信网络的角色正变得愈发关键。 2025年6月6日,为期三天的华南国际工业博览会在深圳国际会展中心(宝安)圆满落幕。作为国内工业通信领域的技术型企业,光路科技(Fiberroad&…...
上位机开发过程中的设计模式体会(1):工厂方法模式、单例模式和生成器模式
简介 在我的 QT/C 开发工作中,合理运用设计模式极大地提高了代码的可维护性和可扩展性。本文将分享我在实际项目中应用的三种创造型模式:工厂方法模式、单例模式和生成器模式。 1. 工厂模式 (Factory Pattern) 应用场景 在我的 QT 项目中曾经有一个需…...
云原生安全实战:API网关Envoy的鉴权与限流详解
🔥「炎码工坊」技术弹药已装填! 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 一、基础概念 1. API网关 作为微服务架构的统一入口,负责路由转发、安全控制、流量管理等核心功能。 2. Envoy 由Lyft开源的高性能云原生…...
虚幻基础:角色旋转
能帮到你的话,就给个赞吧 😘 文章目录 移动组件使用控制器所需旋转:组件 使用 控制器旋转将旋转朝向运动:组件 使用 移动方向旋转 控制器旋转和移动旋转 缺点移动旋转:必须移动才能旋转,不移动不旋转控制器…...
