2020.9.8C++Primer学习笔记————模板函数

2024-03-19 23:44

本文主要是介绍2020.9.8C++Primer学习笔记————模板函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

C++Primer学习笔记————模板函数

看C++Primer看到了第十章函数模板部分,其中提到了模板函数用法,帮助强类型语言减少简单方法的代码量。

C++是强类型语言,在调用方法时需要对传参有严格的判断,例如实现一个简单的大小判断方法时:

int min( int a, int b ) { return a < b ? a : b; 
} 
double min( double a, double b ) { return a < b ? a : b; 
}

想要对比不同类型的参数需要不同的方法,使得一个简单的方法需要重复定义好几次。
而想要一次编写代替多个重复方法,使用宏定义扩展在某些情况下可以实现,但同时也可能导致结果出错,例如:

#include <iostream> 
#define min(a,b) ((a) < (b) ? (a) : (b)) int main() { const int size = 10; int ia[size]; int elem_cnt = 0; int *p = &ia[0]; // 计数数组元素的个数while ( min(p++,&ia[size]) != &ia[size] ) ++elem_cnt; cout << "elem_cnt : " << elem_cnt << "\texpecting: " << size << endl; return 0; 
}

在这个demo中,因为实参操作p++的存在,最后输出的结果会变成
elem_cnt : 5 expecting: 10
实际count次数只有size的一半,因为应用在指针实参 p 上的后置递增操作随每次扩展而被应用了两次,一次是在 a 和 b 的测试中 另一次是在宏的返回值被计算期间。

模板函数:

函数模板提供了一种机制,通过它我们可以保留函数定义和函数调用的语义。在一个程序位置上封装了一段代码,确保在函数调用之前实参只被计算一次,而无需像宏方案那样绕过 C++的强类型检查。
函数模板提供一个种用来自动生成各种类型函数实例的算法,程序员对于函数接口参数和返回类型中的全部或者部分类型进行参数化 parameterize,而函数体保持不变。

上面是摘自C++Primer,总结下来就是将方法的参数类型(而非参数)进行参数化,将 int ,double 等参数类型也当成参数传入方法。
这种函数模板最适合要传入多个同参数类型的参数时,例如:

template <class Type> 
Type min( Type a, Type b ) { return a < b ? a : b; 
} int main() { // ok: int min( int, int ); min( 10, 20 ); // ok: double min( double, double ); min( 10.0, 20.0 );}

也可以有不同的参数类型传入,有几个参数类型就声明几个Type:

template <class TypeA, class TypeB> 
TypeB plus( TypeA a, TypeB b ) { return b + a;
} int main() { // ok: float plus( int, float ); plus( 10, 20.5 ); }

还可以在模板函数中声明模板非类型参数,其代表一个常量表达式,例如:

template <class Type, int size> 
Type min( const Type (&r_array)[size] ) 
{ /* 找到数组中元素最小值的参数化函数 */ Type min_val = r_array[0]; for ( int i = 1; i < size; ++i ) if ( r_array[i] < min_val ) min_val = r_array[i]; return min_val; 
}

其中 size 就是一个 int 类型的常量。

声明冲突:

如果在全局域中声明了与模板参数同名的对象,函数或类型,则该全局名将被隐藏。

typedef double Type; 
template <class Type> 
Type min( Type a, Type b ) 
{ // tmp 类型为模板参数 Type // 不是全局 typedef Type tmp = a < b ? a : b; return tmp; 
}

如上,即模板参数覆盖全局变量。

在函数模板定义中声明的对象或类型不能与模板参数同名,模板类型参数名可以被用来指定函数模板的返回类型。

引入关键字typename:

在函数模板声明中,如传入的参数是一个类的实体:

class PARM{
public:char *name;
...
}PARM Parm;template <class Parm, class U> Parm minus( Parm* array, U value ) 
{ Parm::name * p; // 这是一个指针声明还是乘法 乘法
}

类型Parm要在实例化时才知道他是一个类的实体,Parm::name * p模板函数声明时才是一个指针,对于模板函数内部的定义来说这个 * 代表乘法。
所以这个时候需要引入 typename 关键字来确定变量到底是指针还是一个类内属性:

template <class Parm, class U> Parm minus( Parm* array, U value ) 
{ typename Parm::name * p; // ok: 指针声明
}

函数模板也可以被声明为 inline 或 extern 形式,但是修饰符需要放在函数名前而不是 template 前。

// ok: 关键字跟在模板参数表之后
template <typename Type> inline Type min( Type, Type ); 
// 错误: inline 指示符放置的位置错误
inline 
template <typename Type> Type min( Array<Type>, int );

函数模板实例化:

模板函数的声明在被调用时会隐式的进行模板实例化,根据传入的参数生成独立的函数,过程是隐式的。
用函数实参的类型来决定模板实参的类型和值的过程被称为模板实参推演

template <typename Type, int size> Type min( Type (&p_array)[size] ) { /* ... */ } 
// pf 指向 int min( int (&)[10] ) 
int (*pf)(int (&)[10]) = &min;

指针 pf 被函数模板实例的地址初始化,编译器通过检查 pf 指向的函数的参数类型来决定模板实例的实参。
pf 的类型是指向函数的指针,该函数有一个类型为 int(&)[10]的参数,当 min()被实例化时,该参数的类型决定了 Type 的模板实参的类型和 size 的模板实参的值。Type 的模板实参为 int,size 的模板实参为 10,被实例化的函数是 min(int(&)[10]) ,指针 pf 指向这个模板实例。

在取函数模板实例的地址时,必须能够通过上下文环境为一个模板实参决定一个惟一的类型或值。如果不能决定出这个惟一的类型或值,就会产生编译时刻错误:

template <typename Type, int size> Type min( Type (&r_array)[size] ) { /* ... */ } 
typedef int (&rai)[10]; 
typedef double (&rad)[20]; 
void func( int (*)(rai) ); 
void func( double (*)(rad) ); 
int main() { // 错误: 哪一个 min() 的实例? func( &min ); 
}

这里func()不知道该取 int 型 min 函数的地址还是 double 型函数的地址,需要用一个强制类型转换显式地指出实参的类型则可以消除编译时刻错误:

int main() { // ok: 强制转换指定实参类型func( static_cast< double(*)(rad) >(&min) ); 
}

模板实参推演:

当函数模板被调用时,对函数实参类型的检查决定了模板实参的类型和值,这个过程被称为模板实参推演

函数模板 min()的函数参数是一个引用,它指向了一个 Type 类型的数组,为了匹配函数参数,函数实参必须也是一个表示数组类型的左值。下面的调用是个错误,因为 pval 是 int*类型而不是 int 数组类型的左值:

template <class Type, int size> Type min( Type (&r_array)[size] ) { /* ... */ }void f( int pval[9] ) { // 错误: Type (&)[] != int* int jval = min( pval ); 
}

想正确传入数组参数,我自己尝试的方法:

int f(int pval[9]) {int ia[9];memcpy(ia, pval, 9*(sizeof(int)));int jval = min(ia);return jval;
}

要想成功地进行模板实参推演,函数实参的类型不一定要严格匹配相应函数参数的类型。
下列三种类型转换是允许的:左值转换,限定转换和到一个基类(该基类根据一个类模板实例化而来)的转换。

这篇关于2020.9.8C++Primer学习笔记————模板函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/827604

相关文章

Pandas中统计汇总可视化函数plot()的使用

《Pandas中统计汇总可视化函数plot()的使用》Pandas提供了许多强大的数据处理和分析功能,其中plot()函数就是其可视化功能的一个重要组成部分,本文主要介绍了Pandas中统计汇总可视化... 目录一、plot()函数简介二、plot()函数的基本用法三、plot()函数的参数详解四、使用pl

C#如何调用C++库

《C#如何调用C++库》:本文主要介绍C#如何调用C++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

利用Python打造一个Excel记账模板

《利用Python打造一个Excel记账模板》这篇文章主要为大家详细介绍了如何使用Python打造一个超实用的Excel记账模板,可以帮助大家高效管理财务,迈向财富自由之路,感兴趣的小伙伴快跟随小编一... 目录设置预算百分比超支标红预警记账模板功能介绍基础记账预算管理可视化分析摸鱼时间理财法碎片时间利用财

如何在 Spring Boot 中实现 FreeMarker 模板

《如何在SpringBoot中实现FreeMarker模板》FreeMarker是一种功能强大、轻量级的模板引擎,用于在Java应用中生成动态文本输出(如HTML、XML、邮件内容等),本文... 目录什么是 FreeMarker 模板?在 Spring Boot 中实现 FreeMarker 模板1. 环

Java学习手册之Filter和Listener使用方法

《Java学习手册之Filter和Listener使用方法》:本文主要介绍Java学习手册之Filter和Listener使用方法的相关资料,Filter是一种拦截器,可以在请求到达Servl... 目录一、Filter(过滤器)1. Filter 的工作原理2. Filter 的配置与使用二、Listen

Python的time模块一些常用功能(各种与时间相关的函数)

《Python的time模块一些常用功能(各种与时间相关的函数)》Python的time模块提供了各种与时间相关的函数,包括获取当前时间、处理时间间隔、执行时间测量等,:本文主要介绍Python的... 目录1. 获取当前时间2. 时间格式化3. 延时执行4. 时间戳运算5. 计算代码执行时间6. 转换为指

Python正则表达式语法及re模块中的常用函数详解

《Python正则表达式语法及re模块中的常用函数详解》这篇文章主要给大家介绍了关于Python正则表达式语法及re模块中常用函数的相关资料,正则表达式是一种强大的字符串处理工具,可以用于匹配、切分、... 目录概念、作用和步骤语法re模块中的常用函数总结 概念、作用和步骤概念: 本身也是一个字符串,其中

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

IDEA自动生成注释模板的配置教程

《IDEA自动生成注释模板的配置教程》本文介绍了如何在IntelliJIDEA中配置类和方法的注释模板,包括自动生成项目名称、包名、日期和时间等内容,以及如何定制参数和返回值的注释格式,需要的朋友可以... 目录项目场景配置方法类注释模板定义类开头的注释步骤类注释效果方法注释模板定义方法开头的注释步骤方法注