C++模板介绍
C++ 模板
C++ 模板是一种强大的泛型编程工具,它允许我们编写通用的代码,可以用于处理多种不同的数据类型。模板允许我们在编写代码时将类型作为参数进行参数化,从而实现代码的重用性和灵活性。
在 C++ 中,模板由关键字 template 开始,并且后面跟着模板参数列表。模板参数可以是类型参数或非类型参数。
1、模板的基本语法
1.1 类型模板参数(函数模板)
类型模板参数允许我们在定义模板时指定一个或多个类型参数,这些类型参数可以在模板的定义中使用。例如,下面是一个简单的模板函数示例:
template <typename T>
T add(T a, T b) {
return a + b;
} int main() {
std::cout << add<int>(1,1); // 输出:2 显示指定模板参数类型
std::cout << add(1.1, 2.2); // 输出:3.3 自动推导模板参数类型为浮点型 return 0;
}
在这个例子中,T 是类型模板参数,它代表一个占位符类型。我们可以在模板函数 add 中使用 T 来进行参数和返回类型的声明。当我们调用 add 函数时,编译器会根据传入的实际类型来推断 T 的值。
1.2 非类型模板参数(类模板)
非类型模板参数允许我们在定义模板时指定一个或多个非类型参数,这些参数可以是整数、枚举、指针或引用。非类型参数的值在编译时确定,且在模板的每个实例化中都是常量。例如,下面是一个使用非类型参数的模板类示例:
template <int Size>
class Array {
private:
int data[Size];
public:
// 构造函数
Array() {
for (int i = 0; i < Size; ++i) {
data[i] = i;
}
} void print() const {
for (int i = 0; i < Size; i++) {
std::cout << data[i] << " ";
}
std::cout << std::endl;
}
}; int main() {
Array<5> arr5;
Array<10> arr10;
arr5.print();// 输出:0 1 2 3 4
arr10.print();// 输出:0 1 2 3 4 5 6 7 8 9
return 0;
}
在这个例子中,Size 是非类型模板参数,它代表数组的大小。我们可以在模板类 Array 的定义中使用 Size 来声明数组的大小,并在构造函数中初始化数组。
2、模板作用及优势
模板在 C++ 中具有重要的作用和优势,可以提高代码的重用性和灵活性。
2.1.代码重用:
模板允许我们编写通用的代码,可以处理多种不同的数据类型,而无需为每种类型编写重复的代码。下面是一个模板函数示例,用于计算数组的总和:
#include <iostream> template <typename T, size_t Size>
T sumArray(T (&arr)[Size]) {
T sum = T();
for (size_t i = 0; i < Size; ++i) {
sum += arr[i];
}
return sum;
} int main() {
int intArray[] = {1, 2, 3, 4, 5};
double doubleArray[] = {1.1, 2.2, 3.3, 4.4, 5.5}; int intSum = sumArray(intArray);
double doubleSum = sumArray(doubleArray); std::cout << "Sum of intArray: " << intSum << std::endl;
std::cout << "Sum of doubleArray: " << doubleSum << std::endl; return 0;
}
在这个示例中,我们定义了一个模板函数 sumArray,它可以接受任意类型的数组作为参数,并计算数组的总和。通过使用模板,我们可以在不修改代码的情况下重复使用这个函数,适用于不同类型的数组。
2.2.类型安全:
模板在编译时进行类型检查,可以提供更好的类型安全性。下面是一个模板类示例,用于实现一个简单的栈数据结构:
#include <iostream>
#include <vector> template <typename T>
class Stack {
private:
std::vector<T> stack; public:
void push(const T& item) {
stack.push_back(item);
} T pop() {
if (stack.empty()) {
throw std::runtime_error("Stack is empty");
}
T item = stack.back();
stack.pop_back();
return item;
} bool isEmpty() const {
return stack.empty();
}
}; int main() {
Stack<int> intStack;
intStack.push(10);
intStack.push(20);
intStack.push(30); while (!intStack.isEmpty()) {
std::cout << intStack.pop() << " ";
}
// 输出: 30 20 10 std::cout << std::endl; Stack<std::string> stringStack;
stringStack.push("Hello");
stringStack.push("World"); while (!stringStack.isEmpty()) {
std::cout << stringStack.pop() << " ";
}
// 输出: World Hello return 0;
}
在这个示例中,我们定义了一个模板类 Stack,它可以存储任意类型的元素。通过使用模板,我们可以在编译时检查类型的一致性,并避免将错误类型的元素推入栈中。
3、C++中的模板与 Java中的泛型的异同
C++中的模板和 Java中的泛型都是泛型编程的概念,它们都可以用于编写通用的代码,以便在多个类型上重复使用。
3.1.相同点
都可以使用泛型来编写通用的代码,以便在多个类型上重复使用。
都允许在编译时进行类型检查,以避免在运行时出现类型错误。
都可以使用类型参数来表示通用的类型。
3.2.主要不同点
语法不同:C++中的模板使用template关键字来声明模板参数,而 Java中的泛型使用<>符号来声明类型参数。
支持的类型不同:在C++中,模板可以使用任何可用的类型,包括内置类型和自定义类型,而在 Java中,泛型不能接受基本类型作为类型参数――它只能接受引用类型。这意味着可以定义
List<Integer>,但是不可以定义List<int>。C++中,参数类型不同,实例类型也不同。而在 Java中,不管类型参数是什么,所有实例都是同一类型,并且类型参数会在运行时被抹去。即,尽管在编译时
ArrayList<String>和ArrayList<Integer>是两种类型,但是在运行时只有ArrayList被加载到 JVM中。Java中,在类/接口上声明的泛型,在本类或本接口中即代表某种类型,可以作为非静态属性的类型、非静态方法的参数类型、非静态方法的返回值类型。但是在静态方法中不能使用类的泛型(因为类的泛型在创建对象时,即实例化时才指定,而静态方法要早于对象的创建,此时类的泛型还没指定而静态结构已经需要使用了)。C++中,类型参数可以用于静态方法和静态变量。
.....
综上所述,虽然C++中数模板和 Java中的泛型都是泛型编程的概念,但它们在实现上有很大差异。
参考资料:
【C++基础语法】
https://blog.csdn.net/hxhxhxhxx/article/details/119334165
【C++模板和泛型详解】
https://blog.csdn.net/jj6666djdbbd/article/details/127155728
【C++泛型和 Java泛型的异同】
https://blog.csdn.net/cnds123/article/details/130778765
https://blog.csdn.net/coding_is_fun/article/details/81564512
C++模板介绍的更多相关文章
- Django框架(十一):模板介绍、模板语言、模板继承、HTML转义
1. 模板介绍 1.1 模板的功能 产生html,控制页面上展示的内容.模板文件不仅仅是一个html文件. 模板文件包含两部分内容: 静态内容:css.js.html. 动态内容:用于动态去产生一些页 ...
- puppet的使用:ERB模板介绍
ERB介绍 全称是Embedded RuBy,意思是嵌入式的Ruby,是一种文本模板技术,用过JSP的话,会发现两者语法很像. 我们项目中一般用ERB来产生各模块的配置文件.ERB模板也可以用来产生W ...
- [Vue 牛刀小试]:第十七章 - 优化 Vue CLI 3 构建的前端项目模板(1)- 基础项目模板介绍
一.前言 在上一章中,我们开始通过 Vue CLI 去搭建属于自己的前端 Vue 项目模板,就像我们 .NET 程序员在使用 asp.net core 时一样,我们更多的会在框架基础上按照自己的开发习 ...
- laravel blade模板介绍
第一步:首先写好模板 在sites下新建立一个app.blade.php文件 写入模板如: <!DOCTYPT html><html><head> <titl ...
- phpcmsv9更换模板介绍
先分享下大概的步骤: 1.上传模版文件到服务器:2.在站点管理 里边[模板风格配置]选择新模板:3.设置不同模型对应模板:4.修改现有的栏目,匹配新模板:5.更新栏目缓存.系统缓存,更新HTML静态页 ...
- Instruments模板介绍(更新中...)
第一章.Instruments(仪器)快速入门 Instruments可以用来收集关于一个或者多个系统进程的性能和行为的数据,并跟踪睡着时间产生的数据.它提供了好几个分析模板: Blank 任何类型都 ...
- Win10系列:VC++ Direct3D模板介绍2
(3)CreateDeviceResources函数 CreateDeviceResources函数默认添加在CubeRenderer.cpp源文件中,此函数用于创建着色器和立体图形顶点.接下来分别介 ...
- Win10系列:VC++ Direct3D模板介绍1
Visual Studio为开发Direct3D应用程序提供了便捷的模版,读者可以不必手动去新建Direct3D中所使用到的基础资源,而只需专注于图形的绘制.本小节主要为读者介绍这个模版中用于绘制图形 ...
- discuz模板介绍
1.discuz目录下template为模板目录 模板套系 discuz每套模板,支持不同的风格,而多个风格组成一套套系. 推荐使用复制的方法创建新的风格 (*默认的公共页面静态资源,存储在discu ...
- DataUml Design 教程5-代码模板介绍(甚于T4模板技术)
DataUml Design 代码模板全然基于C#语言来编写的. 不懂写模板的能够请教作者,随时欢迎. 以下是一段模板代码,这段代码能够获取一个类结构的全部信息. <#@ ...
随机推荐
- Java(数组声明创建、初始化、特点、内存分析、边界)
数组的定义 数组是相同类型数据的有序集合 数组描述的是相同类型的若干数据,按照一定先后次序排序组合而成 其中,每一个数据称作一个数组元素,每个数组元素可以通过下标访问它们 1.数组声明和创建 声明 d ...
- vulnhub_jangow
来源 vulnhub:https://www.vulnhub.com/entry/jangow-101,754/ 描述 难度:简单 这在 VirtualBox 而不是 VMware 上效果更好 我这里 ...
- 【技术积累】Python中的NumPy库【一】
NumPy库是什么 NumPy是Python科学计算的核心库之一,用来进行科学计算,数值分析等矩阵运算.主要提供了以下几种功能: 1.多维数组(ndarray)对象,可以进行快速的数值计算和数组操作: ...
- 浏览器输入URL到网页完全呈现的过程
前言 临近计算机网络期末考试, 最近在复习(预习), 写一遍博客讲解加深印象. 浏览器输入URL过程图 浏览器输入 URL 过程: 当用户在网页上输入网址 URL 后, 浏览器会对网址进行 DNS 域 ...
- 微信公众号redirect_uri 参数错误
前期所有准备工作我就不在这里一一叙述了.在这说一下需要注意的事项: 1.如果域名为www开头,记得把www去掉,否则依旧会报这个错误 2.跳转域名必须是授权域名的子集,如:'授权域名为 www.bai ...
- 【python基础】类-继承
编写类时,并非总是要从空白开始.如果要编写的类时另一个现成类的特殊版本,可使用继承.一个类继承另一个类时,它将自动获得另一个类的所有属性和方法 原有的类称为父类,而新类被称为子类.子类继承了其父类的所 ...
- 将Dubbo注册到Nacos,与DubboAdmin的部署
王有志,一个分享硬核Java技术的互金摸鱼侠加入Java人的提桶跑路群:共同富裕的Java人 本文是<从 0 开始学 Dubbo>系列文章中应用篇的番外篇. 在这篇文章中我会和大家一起部署 ...
- Llama2 论文中译版——开放式基础和微调聊天模型
Llama 2:开放式基础和微调聊天模型 写在前头 因为最近一直在使用 LLM 工具,所以在学习 Llama 2:开放式基础和微调聊天模型 这篇论文的期间,顺手将内容翻译了过来. 整片译文是由 Cha ...
- Vue通过v-modal实现子组件通讯
1.在props设置属性value props: { value: { type: Object, default: {} } } 1.设置data数据,接收value,在mounted或create ...
- Node: 将时间戳转换成日期并分组
// 对时间戳按日期进行分组 let moment = require('moment') let timestamp_array = [ 1645059333000, 1613523333000, ...