【C++】非类型模板参数 | array容器 | 模板特化 | 模板为什么不能分离编译

news2024/11/13 17:59:24

目录

一、非类型模板参数

二、array容器

三、模板特化

为什么要对模板进行特化

函数模板特化

补充一个问题

类模板特化

全特化与偏特化

全特化

偏特化

四、模板为什么不能分离编译

为什么

怎么办

五、总结模板的优缺点


一、非类型模板参数

模板参数分两类:类型形参 与 非类型形参。

类型形参:出现在模板参数列表中,跟在class或者typename之类的参数类型名称。

非类型形参:用一个常量作为类 (函数) 模板的一个参数,在类 (函数) 模板中可将该参数当成常量来使用。

为什么需要非类型模板参数呢?我们先来看这样一个场景。

我定义了一个静态栈MyStack:

#define N 100
template<class T>
class MyStack
{
private:
    int arr[N];
    int top;
};
int main() {
    MyStack<int> st;
    return 0;
}

这样实例化出的栈,大小都是100。那假如我想要两个栈,一个大小为100,一个为200,要怎么办呢?

这就是#define所解决不了的问题了。

为此,我们引入了非类型模板参数,来看看它是怎么处理这个问题的:

template<class T,int N>   //非类型模板参数N
class MyStack
{
private:
    int arr[N];
    int top;
};
int main() {
    MyStack<int, 100> st1;
    MyStack<int, 200> st2;
    return 0;
}

这样实例化出的st1,大小为100;st2,大小为200。

注意:

1.整形家族可以作为非类型模板参数,包括char、short、int、long、longlong。最常见的是int。

浮点数、类对象以及字符串是不允许作为非类型模板参数的。

2.非类型的模板参数必须在编译期就能确认结果。

3.非类型模板参数是是常量,是不能修改的。

不信来修改下试试:

template<class T,int N>
class MyStack
{
public:
	void func() {
		N = 200;   //修改N
	}
private:
	int arr[N];
	int top;
};
int main() {
	MyStack<int, 100> st1;
	st1.func();
	return 0;
}

二、array容器

学习了非类型模板参数,我们就可以了解下array容器。这个容器很少用,了解即可。

array,大小固定的数组

相比之前学过的动态数组vector,array的功能就显得有些鸡肋:array有的功能,vector也有;而vector有的功能,array未必有。

一方面,vector是开在堆上的,大小可以变化;而array开在栈上,大小固定。但这也是array的优势,正因为是栈上分配内存,所以比vector的效率更高。

但总的来说,用array的地方,我们往往也能用vector,vector还更好用。所以这个容器很少用。

三、模板特化

模板特化的概念:

在原模板类的基础上,针对特殊类型所进行特殊化的实现方式。

当模板被使用时,编译器会针对特定的类型使用特定的实现,而非使用通用的实现,从而提高代码的效率。

为什么要对模板进行特化

假如我们想要对9和2进行大小比较,现用两种方式比,然而,比较结果却出现了分歧:

template<class T>
bool IfLess(const T& x, const T& y) {
    return x < y;
}
int main() {
    int a = 2, b = 9;
    int* pa = &a, * pb = &b;
    cout << IfLess(a, b) << endl;   
    cout << IfLess(pa, pb) << endl;
    return 0;
}

存进a、b变量里,得到的是正确的结果。而通过传指针的方式,得到的是错误的结果。

这是因为,这里直接比较了指针的大小,而无法对指针先解引用 再比较。

此时,就需要对模板进行特化。我们希望达到的效果是:传指针也能比大小。

模板特化中分为函数模板特化类模板特化

函数模板特化

函数模板的特化步骤:

1.必须要先有一个基础的函数模板

2.关键字template后面接一对空的尖括号<>

3.函数名后跟一对尖括号,尖括号中指定需要特化的类型

4.函数形参表: 必须要和模板函数的基础参数类型完全相同,如果不同编译器可能会报一些奇怪的错误。

如,这里就没有做到完全相同,所以报错了:

应改为:

template<class T>
bool IfLess(T& x,T& y) {
    return x < y;
}
​
template<>
bool IfLess<int*>(int*& x, int*& y) {   //注意形参要和基础参数完全相同!
    return *x < *y;
}

函数模板特化的示例:

#include<iostream>
using namespace std;
​
template<class T>
bool IfLess(T& x,T& y) {
    return x < y;
}
​
template<>
bool IfLess<int*>(int*& x, int*& y) {
    return *x < *y;
}
​
int main() {
    int a = 2, b = 9;
    int* pa = &a, * pb = &b;
    cout << IfLess(a, b) << endl;
    cout << IfLess(pa, pb) << endl;
    return 0;
}

在调用时,函数会根据实参类型,调用对应的函数模板:

补充一个问题

思考:为什么这样写不对?

#include<iostream>
using namespace std;
template<class T>
bool IfLess(const T& x, const T& y) {
    return x < y;
}
​
template<>
bool IfLess<int*>(const int*& x,const int*& y) {  //明明基础参数类型完全一致,为啥报错?
    return *x < *y;
}
int main() {
    ……
}

编译不通过:

这里看似 和基础参数类型完全一致,实际上只是做到了形式上的一致,内涵上却大相径庭。

在基础函数参数中,const修饰形参x、y,保证它们不被改变;那迁移到 特化的参数中,对应的是:const修饰两个指针x、y,使之不被改变。

而const int*& x,const修饰的是指针x指向的内容不受改变,即const修饰 *x。

正确的写法是:int* const& x,让const直接修饰指针x。

📖如果你还是一头雾水,那说明“指针常量and常量指针”的知识点你没掌握好,现在我予以补充。

指针常量:int* const p。const直接修饰指针p,p自身的值不能改变,即p的指向是不能变的,而*p可以改变。

p就相当于一个常量了,此常量的类型为指针,因而叫指针常量。

常量指针:const int* p或int const* p,const修饰的是*p,即指针指向的内容是不能变的,而指针的指向可以改变。

因为指向的内容不能变,就相当于指向了一个常量,所以叫常量指针。

类模板特化

当我们需要针对某些特定类型进行特殊处理时,就需要对类模板进行特化。

如,我们定义了一个类模板 用于计算两个数的和:

#include <iostream>
using namespace std;
​
template<class T1,class T2>
class Adder
{
public:
    T1 add(T1 x, T2 y) {
        return x + y;
    }
};
​
int main() {
    Adder<int,double> a;
    int ret1 = a.add(1, 1.1);   //我想要ret1是整形,ret2是浮点型
    double ret2 = a.add(1, 1.1);
    cout << ret1 << endl;
    cout << ret2 << endl;
​
    return 0;
}

我想要1+1.1的结果,一个取整为2,一个保留小数为2.1。然而,结果却:

结果都是整形。因为返回类型T1是int,这会直接把2.1截断,返回2,存进ret2里。

来看看 当引入类模板特化,是怎么解决这个问题的吧:

#include <iostream>
using namespace std;
​
template<class T1,class T2>
class Adder
{
public:
    T1 add(T1 x, T2 y) {
        return x + y;
    }
};
​
template<>
class Adder<int,double>
{
public:
    double add(int x, double y) {
        return x + y;
    }
    
};
​
int main() {
    Adder<int,double> a;
    int ret1 = a.add(1, 1.1);
    double ret2 = a.add(1, 1.1);
    cout << ret1 << endl;
    cout << ret2 << endl;
​
    return 0;
}

所以说,当需要针对特殊情况做特殊处理时,可以考虑使用类模板特化。

全特化与偏特化

全特化

将所有的模板参数都确定化。(我们刚刚给出的那几个例子,都是全特化的)

例:

template<class T1,class T2>
class Adder
{
public:
    T1 add(T1 x, T2 y) {
        return x + y;
    }
};
​
template<>
class Adder<int,double>
{
public:
    double add(int x, double y) {
        return x + y;
    }
};

注意看,全特化时,template<>尖括号里一定是空的。在尖括号内不添加类型,表示完全特化。

(但不能凭<>是否为空,来区分是全or偏特化。偏特化的<>也可以为空)

偏特化

偏特化又叫半特化。下面这2种情况都属于偏特化:

1.将参数表中一部分参数进行特化。

例:

template<class T1,class T2>
class Adder
{
public:
    T1 add(T1 x, T2 y) {
        return x + y;
    }
};
​
template<class T1>   //只特化了T2,T1还得保留
class Adder<T1,double>
{
public:
    double add(T1 x, double y) {
        return x + y;
    }
    
};

2.对参数进行更进一步的限制。

例:如果我们想要 传参时,确保是引用传参,该怎么做呢?

template<class T1, class T2>
class Adder
{
public:
    T1 add(T1 x, T2 y) {
        cout << "调用了普通类模板" << endl;
        return x + y;
    }
};
​
template<class T1,class T2>
class Adder<T1& ,T2&>    //限制两个参数是引用
{
public:
    double add(const T1& x, const T2& y) {
        cout << "调用了特化类模板" << endl;
        return x + y;
    }
​
};
int main() {
    Adder<int,double> a;
    a.add(1, 1.1);
​
    Adder<int&, double&> b;  //调用特化的类模板,这样的话,就一定是传引用传参
    b.add(1, 1.1);
​
    return 0;
}

四、模板为什么不能分离编译

为什么

最开始接触模板时,就说过:模板不能分离编译。那为什么呢?我们要知其然,还要知其所以然。现在,我通过一个例子来说明这个原因。

我现在在一个project里创建了三个文件:Add.h、Add.cpp、main.cpp,分别用于声明、定义和测试。

//Add.h
#include<iostream>
using namespace std;
template<class T1,class T2>
T1 Add(T1 x,T2 y);
​
//Add.cpp
#include"Add.h"
template<class T1, class T2>
T1 Add(T1 x, T2 y) {
    return x + y;
}
​
//main.cpp
#include"Add.h"
int main() {
    cout << Add(1, 2.0) << endl;
    return 0;
}

运行报错:

原因说明:

我们先来回顾下,程序是怎么运行的。

程序在运行时,要经历四个阶段:预处理、编译、汇编、链接。

a.在预处理阶段,头文件被展开、宏被替换、注释被删除;

b.在编译阶段,编译器对各个源文件分别进行语法检查,然后将其转化成汇编代码。(注意:此阶段已经没有头文件了)

c.由于机器只能识别0、1串,所以我们的代码要经过汇编,从给人看的字符,变成给机器看的01。在汇编阶段,会形成符号表(用于存放变量、函数的地址)。

d.链接时,计算机遇到不认识的变量、函数,就会去符号表里找它的地址。通过链接,一个项目里的多个文件,才能合成一个关联的整体。

模板之所以不能分离编译,就是在链接阶段出岔子了。

我们说过,模板就像一张图纸。Add.cpp里的函数模板,因为没有确定的类型,所以并未实例化出具体的函数,自然无法在符号表中生成对应的地址。

在mian.cpp中,模板隐式实例化出 函数Add<int,double>,想要调用,却在符号表里找不到地址。所以报错:Add<int,double>为无法解析的外部符号。

怎么办

既然不能分离编译,那一般怎样处理模板呢?这里提供两种方法。

1.(推荐!)用到模板的地方,就不要分离编译~

将声明和定义放到同一个文件 "xxx.hpp" (或者"xxx.h")。

2.(不推荐)模板定义的位置显式实例化。

你要调用的函数,都显式实例化放在Add.cpp中:

template<class T1, class T2>
T1 Add(T1 x, T2 y) {
    return x + y;
}
//显示实例化
template
int Add(int x, double y);

这样写不好,因为如果你要调用好几种不同参数类型的Add,你就都要显式实例化,这就造成了代码的冗余。

五、总结模板的优缺点

优点:

1.模板复用了代码,节省资源,更快的迭代开发,C++的标准模板库(STL)因此而产生

2.增强了代码的灵活性

缺点:

1.模板会导致代码膨胀问题,也会导致编译时间变长

2.出现模板编译错误时,错误信息非常凌乱,不易定位错误

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MVVM框架:图片加载有问题

一、前言&#xff1a;在我使用ImageView加载图片的时候添加如下代码发现报错 app:imageUrl"{viewModel.observableField.assetImg}"报错如下错误 二、原因&#xff1a;是啥我不太清楚好像是没有imageView的适配器&#xff0c;后来我看了一下确实没有 public class I…

Java中所有的运算符,以及运算符优先级(总结)

运算法是一种特殊的符号&#xff0c;用于表示数据的运算、复制、比较等。 1、算数运算符 // % 取余运算&#xff1a;结果的符号和被模数的符号一致 12 % 5 2 -12 % 5 -2 12 % -5 2 -12 % -5 -2int a1 10; int b1 a1; // a111, b111 int a2 10; int b2 a2; // a211, …

keras转onnx,TensorFlow转tf.keras.models.load_model,onnx精度转换

参考&#xff1a; https://blog.csdn.net/Deaohst/article/details/126864267 转onnx 别直接转onnx。 先转PB&#xff1a; import tensorflow as tfmodel_path ./models/model.h5 # 模型文件 model tf.keras.models.load_model(model_path) model.sa…

Microsoft outlook已停止工作

故障现象 运行outlook&#xff0c;无法进入程序,显示已停止工作 故障截图 解决方案 1、点击查看问题详细信息&#xff0c;查看故障模块名称&#xff0c;一般是某个dll动态链文件&#xff0c;然后下载或者拷贝该dll文件将其覆盖即可 2、如果找不到&#xff0c;可以使用专用的…

如何向MapInfo Pro添加自定义符号?

用户可以在MapInfo Pro中创建和使用自己的自定义图像作为符号。要访问这些自定义符号&#xff0c;请将它们放在CUSTSYMB目录中&#xff0c;然后从“符号样式”对话框&#xff08;Style>符号样式&#xff09;的“字体”列表中的“自定义符号”选项中选择它们。MapInfo Pro中的…

智慧化城市内涝的预警,万宾科技内涝积水监测仪

随着城市化进程的加速&#xff0c;伴随的是城市内涝问题日益凸显。频繁的暴雨和积水给市民的生活带来了诸多不便&#xff0c;也给城市的基础设施带来了巨大压力。如何解决这一问题&#xff0c;成为智慧城市建设的重要课题和政府管理的工作主题&#xff0c;只要内涝问题得到缓解…

实时数仓-Flink使用总结

阿里云实时计算Flink版是阿里云基于Apache Flink构建的企业级、高性能实时大数据处理系统。具备一站式开发运维管理平台&#xff0c;支持作业开发、数据调试、运行与监控、自动调优、智能诊断等全生命周期能力。本期将对Flink的使用进行总结。 1. Flink产品回顾 阿里云实时计算…

11月13日星期一今日早报简报微语报早读

11月13日星期一&#xff0c;农历十月初一&#xff0c;早报微语早读。 1、国家邮政局&#xff1a;“双11”当天全国快递业务量达6.39亿件&#xff1b; 2、公安机关通缉4名缅北电诈头目&#xff0c;其中一人为缅甸掸邦议会原议员&#xff1b; 3、多部门提醒&#xff1a;未满10…

【Debug】此语法需要一个导入的帮助程序,但找不到模块“tslib”

报错如下&#xff1a; 代码可以编译运行&#xff0c;但是会有红线和报错。 解决方法&#xff1a;TypeScript error “TS2354: This syntax requires an imported helper but module ‘tslib’ cannot be found” Issue #37991 microsoft/TypeScript GitHub 在tsconfig.jso…

MacOS下VMware Fusion配置静态IP

前言 在虚拟机安装系统后&#xff0c;默认是通过DHCP动态分配的IP&#xff0c;这会导致每次重启虚拟机ip都可能会改变&#xff0c;使用起来会有很多不便。 配置静态IP 查看主机网关地址 cat /Library/Preferences/VMware\ Fusion/vmnet8/nat.conf 查看主机DNS&#xff0c;m…

【中间件篇-Redis缓存数据库06】Redis主从复制/哨兵 高并发高可用

Redis高并发高可用 复制 在分布式系统中为了解决单点问题&#xff0c;通常会把数据复制多个副本部署到其他机器&#xff0c;满足故障恢复和负载均衡等需求。Redis也是如此&#xff0c;它为我们提供了复制功能&#xff0c;实现了相同数据的多个Redis 副本。复制功能是高可用Re…

贵金属投资怎样分析市场行情?

分析市场行情对于贵金属投资者来说非常重要&#xff0c;它可以帮助投资者更好地理解市场趋势和价格走势&#xff0c;从而做出更明智的投资决策。以下是一些常用的方法和工具&#xff0c;可用于贵金属市场行情分析&#xff1a; 1、基本面分析&#xff1a;基本面分析主要关注供求…

数字货币swap交易所逻辑系统开发分析方案

随着数字货币市场的快速发展&#xff0c; Swap交易所已成为一种重要的交易方式。本文将对数字货币Swap交易所逻辑系统开发进行分析&#xff0c;并探讨其优势、开发难点和解决方案。 一、数字货币Swap交易所逻辑系统开发的优势 数字货币Swap交易所是一种点对点的交易方式&#x…

更好的运用重生奇迹mu的每一个石头

重生奇迹mu中的每一个石头都是非常起作用的&#xff0c;然而任何一个石头&#xff0c;只要运用好了&#xff0c;在关键时刻&#xff0c;都是能够对我们起到非常直接的作用的&#xff0c;尽可能的把石头运用在最关键的细节上&#xff0c;那么这样能够达到的效果&#xff0c;也就…

压测必经之路,Jmeter分布式压测教程!

01、分布式压测原理 Jemter分布式压测是选择其中一台作为调度机&#xff08;master&#xff09;&#xff0c;其他机器作为执行机&#xff08;slave&#xff09;&#xff1b;当然一台机器也可以既做调度机&#xff0c;也做执行机。 调度机执行脚本的时候&#xff0c;master将会…

OpenWRT搭建个人web站点并结合内网穿透实现公网远程访问

文章目录 前言1. 检查uhttpd安装2. 部署web站点3. 安装cpolar内网穿透4. 配置远程访问地址5. 配置固定远程地址 前言 uhttpd 是 OpenWrt/LuCI 开发者从零开始编写的 Web 服务器&#xff0c;目的是成为优秀稳定的、适合嵌入式设备的轻量级任务的 HTTP 服务器&#xff0c;并且和…

231030期就业班开班咯!我在前方护航,让你稳稳入职

就业哪家强&#xff1f;还得看优橙! 11月9日&#xff0c;231030期就业班的小伙伴结束了为期8天的基础班学习&#xff0c;正式进入了就业班。优橙教育也为新一批就业班的同学举办了开班典礼。 典礼环节中不仅有多彩的抽奖活动&#xff0c;也有丰富的超值礼品&#xff0c;旨在鼓…

如何使用软路由R4S+iStoreOS实现远程访问内网电脑桌面

软路由R4SiStoreOS实现公网远程桌面局域网内电脑 文章目录 软路由R4SiStoreOS实现公网远程桌面局域网内电脑简介一、配置远程桌面公网地址二、家中使用永久固定地址 访问公司电脑**具体操作方法是&#xff1a;** 简介 上篇教程我们介绍了如何在iStoreOS中安装Cpolar&#xff0…

uniapp——项目day03

商品列表 分支创建 定义请求参数对象 获取商品列表数据 渲染商品列表结构 1. 在页面中&#xff0c;通过 v-for 指令&#xff0c;循环渲染出商品的 UI 结构&#xff1a; <template><view><view class"goods-list"><block v-for"(goods,…

响应式珠宝首饰展示网站模板源码带后台

模板信息&#xff1a; 模板编号&#xff1a;6201 模板编码&#xff1a;UTF8 模板颜色&#xff1a;黑白 模板分类&#xff1a;服饰、箱包、礼品、玩具 适合行业&#xff1a;珠宝饰品类企业 模板介绍&#xff1a; 本模板自带eyoucms内核&#xff0c;无需再下载eyou系统&#xf…