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 = int
    std::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 = int
    Box<double> doubleBox(3.14);   // T = double
    Box<std::string> strBox("Hi"); // T = std::string

    std::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 = 5
    Array<double, 10> arr2; // T = double, N = 10

    std::cout << arr1.size() << std::endl; // 输出 5
    std::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);  // 输出:普通值: 42

    Box<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() 没有参数,终止递归。


✨ 总结

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

使用linux脚本部署discuz博客(详细注释版)

使用脚本部署一个discuzz项目 1.显示当前环境状态 防火墙状态 selinux状态 httpd状态 由上可知&#xff0c;虚拟机已处于最初始状态 2.脚本编写 #!/bin/bash #这是一个通过脚本来部署discuzz博客 firewalld关闭 systemctl stop firewalld if [ $? -eq 0 ];then echo "…

【代码审计】-Tenda AC 18 v15.03.05.05 /goform接口文档漏洞挖掘

路由器&#xff1a;Tenda AC 18 v15.03.05.05 固件下载地址&#xff1a;https://www.tenda.com.cn/material?keywordac18 1./goform/SetSpeedWan 接口文档&#xff1a; formSetSpeedWan函数中speed_di参数缓冲区溢出漏洞&#xff1a; 使用 binwalk -eM 解包固件&#xff0c…

正式页面开发-登录注册页面

整体路由设计&#xff1a; 登录和注册的切换是切换组件或者是切换内容&#xff08;v-if和 v-else)&#xff0c;因为点击两个之间路径是没有变化的。也就是登录和注册共用同一个路由。登录是独立的一级路由。登录之后进到首页&#xff0c;有三个大模块&#xff1a;文章分类&…

Unity 位图字体

下载Bitmap Font Generator BMFont - AngelCode.com 解压后不用安装直接双击使用 提前设置 1、设置Bit depth为32 Options->Export options 2、清空所选字符 因为我们将在后边导入需要的字符。 Edit->Select all chars 先选择所有字符 Edit->Clear all chars i…

双重差分学习笔记

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

项目设置内网 IP 访问实现方案

在我们平常的开发工作中&#xff0c;项目开发、测试完成后进行部署上线。比如电商网站、新闻网站、社交网站等&#xff0c;通常对访问不会进行限制。但是像企业内部网站、内部管理系统等&#xff0c;这种系统一般都需要限制访问&#xff0c;比如内网才能访问等。那么一个网站应…

数仓搭建(hive):DWB层(基础数据层)

维度退化: 通过减少表的数量和提高数据的冗余来优化查询性能。 在维度退化中&#xff0c;相关的维度数据被合并到一个宽表中&#xff0c;减少了查询时需要进行的表连接操作。例如&#xff0c;在销售数据仓库中&#xff0c;客户信息、产品信息和时间信息等维度可能会被合并到一…

多模态特征提取与融合助力高光谱+LiDAR数据分类性能飞跃

目录 论文解读 总体架构 CMIIE 模块工作模式 MLFFC模块工作模式 论文解读 提出了一种新的多模态特征提取模块CMIIE,可以捕获高光谱和LiDAR数据之间的互补信息。设计了一个多层特征融合分类模块MLFFC,通过对不同层级的特征进行融合来提高分类性能。使用对抗学习策略来指导网…

Flutter 正在推进全新 PlatformView 实现 HCPP, 它又用到了 Android 上的什么黑科技

跨平台开发里的 PlatformView 实现一直是一个经久不衰的话题&#xff0c;在之前的 《深入 Flutter 和 Compose 的 PlatformView 实现对比》 我们就详细聊过 Flutter 和 Compose 在 PlatformView 实现上的异同之处&#xff0c;也聊到了 Compose 为什么在相同实现上对比 Flutter …

Qt/C++面试【速通笔记一】

Qt 信号与槽机制 什么是信号&#xff08;Signal&#xff09;和槽&#xff08;Slot&#xff09;&#xff1f; 在Qt中&#xff0c;信号&#xff08;Signal&#xff09;和槽&#xff08;Slot&#xff09;是实现对象之间通信的一种机制。信号是对象在某些事件发生时发出的通知&…

《跟李沐学 AI》AlexNet论文逐段精读学习心得 | PyTorch 深度学习实战

前一篇文章&#xff0c;使用 AlexNet 实现图片分类 | PyTorch 深度学习实战 本系列文章 GitHub Repo: https://github.com/hailiang-wang/pytorch-get-started 本篇文章内容来自于学习 9年后重读深度学习奠基作之一&#xff1a;AlexNet【下】【论文精读】】的心得。 《跟李沐…

【科研绘图系列】R语言绘制小提琴图、散点图和韦恩图(violin scatter plot Venn)

禁止商业或二改转载,仅供自学使用,侵权必究,如需截取部分内容请后台联系作者! 文章目录 介绍加载R包数据下载画图1画图2画图3画图4画图5画图6画图7参考介绍 【科研绘图系列】R语言绘制小提琴图、散点图和韦恩图(violin & scatter plot & Venn) 加载R包 library…

IMX6ULL的ALT0、ALT1、ALT2、ALT3、ALT4等是啥意思?

在IMX6ULL的手册IMX6ULLRM.pdf中&#xff0c;发现了题目中这些描述&#xff0c;相关截图如下&#xff1a; 那么红框中的ALT0、ALT1、ALT2、ALT3、ALT4等是啥意思呢&#xff1f; 在IMX6ULL及其他NXP&#xff08;Freescale&#xff09;芯片中&#xff0c;ALT0、ALT1、ALT2、ALT…

Android Http-server 本地 web 服务

时间&#xff1a;2025年2月16日 地点&#xff1a;深圳.前海湾 需求 我们都知道 webview 可加载 URI&#xff0c;他有自己的协议 scheme&#xff1a; content:// 标识数据由 Content Provider 管理file:// 本地文件 http:// 网络资源 特别的&#xff0c;如果你想直接…

DeepSeek 冲击(含本地化部署实践)

DeepSeek无疑是春节档最火爆的话题&#xff0c;上线不足一月&#xff0c;其全球累计下载量已达4000万&#xff0c;反超ChatGPT成为全球增长最快的AI应用&#xff0c;并且完全开源。那么究竟DeepSeek有什么魔力&#xff0c;能够让大家趋之若鹜&#xff0c;他又将怎样改变世界AI格…

神经网络八股(1)

1.什么是有监督学习&#xff0c;无监督学习 有监督学习是带有标签的&#xff0c;无监督学习是没有标签的&#xff0c;简单来说就是有监督学习的输入输出都是固定的&#xff0c;已知的&#xff0c;无监督学习输入是已知的&#xff0c;输出是不固定的&#xff0c;无监督学习是通…

DeepSeek 助力 Vue 开发:打造丝滑的瀑布流布局(Masonry Layout)

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 Deep…

【分布式理论14】分布式数据库存储:分表分库、主从复制与数据扩容策略

文章目录 一、分表分库1. 数据分表的必要性与方式2. 数据分库原则与优势 二、主从复制1. 读写分离架构设计2. 数据复制方式3. MySQL实现主从复制4. MySQL主从复制实践与高可用方案 三、数据扩容 随着业务的不断发展和数据量的增长&#xff0c;传统的单机关系型数据库已经逐渐不…

从传统到轻量级5G:网络架构演变与优化路径

轻量级5G​​​​ 随着5G技术的不断发展&#xff0c;通信网络架构正经历着前所未有的变革。传统的5G核心网架构虽然在性能和容量方面表现出色&#xff0c;但在灵活性、部署效率以及成本控制方面却面临一些挑战。为了应对日益复杂的通信需求&#xff0c;轻量级5G核心网成为了一种…

搭建Kubernetes (K8s) 集群----Centos系统

前期准备 准备3台Linux虚拟机&#xff08;CentOS系统&#xff09;&#xff0c;参考 https://carry.blog.csdn.net/article/details/144578009https://carry.blog.csdn.net/article/details/144578009搭建Docker环境&#xff0c;参考 https://carry.blog.csdn.net/article/de…