C++ 模板 简单易懂

简单易懂的比喻+代码示例+逐步递进,让你学得轻松又扎实!


1️⃣ 模板(Template)是什么?

🌟 先来个简单的例子

假设你写了一个 计算两个数之和 的函数:

int add(int a, int b) {return a + b;
}

可以处理整数 int,但如果想计算 小数 double 呢?
你得再写一个函数

double add(double a, double b) {return a + b;
}

⚠️ 问题: 代码重复!不同类型的数据都要单独实现,怎么办?
💡 模板 来解决!


2️⃣ 函数模板

🚀 基本语法

template <typename T>
T add(T a, T b) {return a + b;
}
🔹 代码解析
  • template <typename T>:告诉编译器 T 是一个占位符,代表"某种类型"。
  • T add(T a, T b):参数 ab 和返回值的类型都是 T,编译时会替换成具体类型

🌰 示例:使用模板

#include <iostream>// 定义函数模板
template <typename T>
T add(T a, T b) {return a + b;
}int main() {std::cout << add(3, 5) << std::endl;      // T = intstd::cout << add(3.14, 2.71) << std::endl; // T = double
}

🎯 编译器会自动推导 T 的类型

  • add(3, 5)int add(int a, int b)
  • add(3.14, 2.71)double add(double a, double b)

3️⃣ 类模板

🚀 为什么要用类模板?

问题: 你写了一个 存储整数Box 类:

class Box {
private:int value;
public:Box(int v) : value(v) {}int getValue() { return value; }
};

但如果想存 doublestd::string 呢?
类模板一个类 适用于 所有类型

🌟 语法

template <typename T>
class Box {
private:T value;
public:Box(T v) : value(v) {}T getValue() { return value; }
};

🌰 使用类模板

#include <iostream>// 定义类模板
template <typename T>
class Box {
private:T value;
public:Box(T v) : value(v) {}T getValue() { return value; }
};int main() {Box<int> intBox(10);           // T = intBox<double> doubleBox(3.14);   // T = doubleBox<std::string> strBox("Hi"); // T = std::stringstd::cout << intBox.getValue() << std::endl;std::cout << doubleBox.getValue() << std::endl;std::cout << strBox.getValue() << std::endl;
}

🎯 T 被替换成不同类型,一个模板搞定所有情况!


4️⃣ 非类型模板参数

🚀 什么是非类型模板参数?

模板不仅可以接受类型参数,还可以接受常量参数!

template <typename T, int N>
class Array {
private:T data[N];  // N 是数组大小
public:int size() { return N; }
};

🌰 示例

#include <iostream>// 定义类模板,N 是数组大小
template <typename T, int N>
class Array {
private:T data[N];
public:int size() { return N; }
};int main() {Array<int, 5> arr1;   // T = int, N = 5Array<double, 10> arr2; // T = double, N = 10std::cout << arr1.size() << std::endl; // 输出 5std::cout << arr2.size() << std::endl; // 输出 10
}

🎯 N 是常量,必须在编译时确定,不能传变量


5️⃣ 模板特化

🚀 什么是模板特化?

问题: 如果 Box<std::string> 需要特殊处理,怎么办?
模板特化(Template Specialization) 允许我们针对特定类型提供不同实现

🌰 示例

#include <iostream>// 普通模板
template <typename T>
class Box {
public:void print(T value) {std::cout << "普通值: " << value << std::endl;}
};// 针对 std::string 特化
template <>
class Box<std::string> {
public:void print(std::string value) {std::cout << "字符串: " << value << std::endl;}
};int main() {Box<int> intBox;intBox.print(42);  // 输出:普通值: 42Box<std::string> strBox;strBox.print("Hello");  // 输出:字符串: Hello
}

🎯 特化后的 Box<std::string> 代码与普通模板不同,实现了不同逻辑


6️⃣ 可变参数模板

🚀 为什么需要可变参数模板?

问题: 有时候,函数/类的参数个数不固定,怎么办?
可变参数模板(Variadic Template) 允许任意数量的模板参数

🌰 示例

#include <iostream>// 递归终止条件
void print() { std::cout << std::endl; }// 递归展开参数
template <typename T, typename... Args>
void print(T first, Args... rest) {std::cout << first << " ";print(rest...);  // 递归展开剩余参数
}int main() {print(1, "Hello", 3.14, "C++", 42);
}

🎯 编译器会递归展开参数,直到 print() 没有参数,终止递归。


✨ 总结

特性作用
函数模板适用于 任意类型 的函数
类模板适用于 任意类型 的类
非类型模板参数模板参数不仅限于类型,还可以是整数常量
模板特化针对特定类型 提供不同实现
可变参数模板允许 任意数量的模板参数

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/21300.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

nlp|微调大语言模型初探索(3),qlora微调deepseek记录

前言 上篇文章记录了使用lora微调llama-1b,微调成功,但是微调llama-8b显存爆炸,这次尝试使用qlora来尝试微调参数体量更大的大语言模型,看看64G显存的极限在哪里。 1.Why QLora? QLoRA 在模型加载阶段通过 4-bit 量化大幅减少了模型权重的显存占用。QLoRA 通过 反量化到 …

【设计模式】【创建型模式】工厂方法模式(Factory Methods)

&#x1f44b;hi&#xff0c;我不是一名外包公司的员工&#xff0c;也不会偷吃茶水间的零食&#xff0c;我的梦想是能写高端CRUD &#x1f525; 2025本人正在沉淀中… 博客更新速度 &#x1f44d; 欢迎点赞、收藏、关注&#xff0c;跟上我的更新节奏 &#x1f3b5; 当你的天空突…

DeepSeek模型快速部署教程-搭建自己的DeepSeek

前言&#xff1a;在人工智能技术飞速发展的今天&#xff0c;深度学习模型已成为推动各行各业智能化转型的核心驱动力。DeepSeek 作为一款领先的 AI 模型&#xff0c;凭借其高效的性能和灵活的部署方式&#xff0c;受到了广泛关注。无论是自然语言处理、图像识别&#xff0c;还是…

Deepseek 与 ChatGPT:AI 浪潮中的双子星较量

引言 在人工智能飞速发展的当下&#xff0c;AI 语言模型成为了人们关注的焦点。Deepseek 与 ChatGPT 作为其中的佼佼者&#xff0c;各自展现出独特的魅力&#xff0c;引领着 AI 技术的发展潮流。今天&#xff0c;就让我们深入探讨这两款模型&#xff0c;看看它们在 AI 领域中是…

QT事件循环

文章目录 主事件循环事件循环事件调度器事件处理投递事件发送事件 事件循环的嵌套线程的事件循环deleteLater与事件循环QEventLoop类QEventLoop应用等待一段时间同步操作模拟模态对话框 参考 本文主要对QT中的事件循环做简单介绍和使用 Qt作为一个跨平台的UI框架&#xff0c;其…

解决DeepSeek服务器繁忙问题的实用指南

目录 简述 1. 关于服务器繁忙 1.1 服务器负载与资源限制 1.2 会话管理与连接机制 1.3 客户端配置与网络问题 2. 关于DeepSeek服务的备用选项 2.1 纳米AI搜索 2.2 硅基流动 2.3 秘塔AI搜索 2.4 字节跳动火山引擎 2.5 百度云千帆 2.6 英伟达NIM 2.7 Groq 2.8 Firew…

进程等待和进程程序替换

进程控制 进程等待进程程序替换 进程等待 如果子进程没有退出 而父进程在进行执行waitpid进行等待&#xff0c;阻塞等待&#xff0c; 进程阻塞了 在等待某种条件发生&#xff08;子进程退出&#xff09; 进程程序替换 1 #include <stdio.h>2 #include <unistd.h>3…

UEFI Spec 学习笔记---6 - Block Translation Table (BTT) Layout

6.1 Block Translation Table (BTT) Background 定义个一个连续地址的非易失性的namespace&#xff0c;就是将一整个namespace 拆分成一个个block&#xff0c;其中的地址保存至BBT&#xff08;块转换表&#xff09;&#xff0c;这样可以防止扇区撕裂&#xff08;由于电源问题导…

SAP 代码扫描工具

描述&#xff1a; ZSCANNER是一个先进的代码分析工具&#xff0c;旨在提供对程序和功能模块内部工作的全面见解。它揭示了代码的技术细节&#xff0c;包括正在创建、读取、更新或删除的数据表&#xff08;CRUD操作&#xff09;&#xff0c;以及正在调用的类、功能模块和BAPI&a…

c语言基础第12节《函数的调用》

c语言基础10 函数 函数的调用 调用方式 ①函数语句&#xff1a; test(); // 对于无返回值的函数&#xff0c;直接调用 int res max(2,4); // 对于有返回值的函数&#xff0c;一般需要再主调函数中接收被调函数的返回值。②函数表达式&#xff1a; 4 max(2,4) scanf(&qu…

C++:iterator迭代器失效

说明&#xff1a;这里拿 vector 举例。 原因一&#xff1a;扩容导致迭代器失效 // 迭代器失效 void insert(iterator pos, const T& x) {assert(pos > _start);assert(pos < _finish);// 满了就扩容if (_finish _end_of_storage){reserve(capacity() 0 ? 4 : ca…

QT之改变鼠标样式

QT改变鼠标图片 资源路径如下 代码实现 QPixmap customCursorPixmap(":/images/mouse.png");QCursor customCursor(customCursorPixmap);QWidget::setCursor(customCursor); // 可以设置为整个窗口或特定控件QWidget::setCursor(); // 设置为透明光标&#xff0c…

用DeepSeek零基础预测《哪吒之魔童闹海》票房——从数据爬取到模型实战

系列文章目录 1.元件基础 2.电路设计 3.PCB设计 4.元件焊接 5.板子调试 6.程序设计 7.算法学习 8.编写exe 9.检测标准 10.项目举例 11.职业规划 文章目录 **一、为什么要预测票房&#xff1f;****二、准备工作****三、实战步骤详解****Step 1&#xff1a;数据爬取与清洗&am…

高并发下秒杀系统的设计

文章目录 1 业界通用做法1.1 压力分摊1.2 RedisMySQL1.3 Inventory Hint1.4 压力分摊RedisMQ 2 Redis MQ 解决高并发下的秒杀场景2.1 Redis库存预扣减2.1.1 lua脚本执行流程&#xff1a;2.1.2 Lua脚本主要做了几件事&#xff1a; 2.2 MySQL库存扣减2.3 记录操作流水的原因 3 I…

双重差分学习笔记

双重差分适用的研究场景&#xff1a; 研究某项政策或者冲击造成的影响 例如&#xff0c;某某小学在2024.12.12日颁布了小红花激励措施&#xff0c;我们要研究这项措施对学生成绩的影响&#xff0c;此时&#xff0c;就可以使用双重差分模型。 双重差分适用的数据类型&#xf…

深入理解 C++17 中的 std::atomic<T>::is_always_lock_free

文章目录 原子操作与锁无关性&#xff08;Lock-Free&#xff09;锁无关性&#xff08;Lock-Free&#xff09;无锁&#xff08;Lock-Free&#xff09;与无阻塞&#xff08;Wait-Free&#xff09; std::atomic<T>::is_always_lock_free 是什么&#xff1f;truefalse与 is_l…

VSCode 中 Git 添加了多个远端,如何设置默认远端

VSCode 中 Git 添加了多个远端&#xff0c;如何设置默认远端 查看分支&#xff1a;设置默认远端手动指定远端 查看分支&#xff1a; * 表示当前默认远端 git branch -vv* master a1b2c3d [origin/main] Fix typo dev d4e5f6g [upstream/dev] Add feature设置默认远端 将本…

一文讲清 AIO BIO NIO的区别

引言 在 Java 编程中&#xff0c;BIO&#xff08;Blocking I/O&#xff09;、NIO&#xff08;Non-blocking I/O&#xff09;和 AIO&#xff08;Asynchronous I/O&#xff09;是三种不同的 I/O 模型&#xff0c;它们在处理输入输出操作时有着不同的机制和特点&#xff0c;但是市…

使用(xshell+xftp)将前端项目部署到服务器

一.以vue项目为例 将项目打包生成dist文件 二.下载载安装xshell和xftp 下载地址&#xff1a;家庭/学校免费 - NetSarang Website 三.连接服务器 在xshell新建会话&#xff08;需要用到服务器、用户名、密码、端口号&#xff09;正确输入后连接到服务器 使用命令连接&#x…

硬件岗位是否适合你?

在当今科技飞速发展的时代,硬件行业作为技术创新的基石,始终扮演着至关重要的角色。无论是智能手机、自动驾驶汽车,还是人工智能服务器,硬件都是这些技术的核心支撑。然而,硬件岗位是否适合你?作为一名硬件专家,我将从多个角度为你分析,帮助你判断自己是否适合从事硬件…