26、C++ Primer 4th 笔记,模板与泛型编程(1)

简介: 1、所谓泛型编程就是以独立于任何特定类型的方式编写代码。使用泛型程序时,我们需要提供具体程序实例所操作的类型或值。     在泛型编程中,我们所编写的类和函数能够多态地用于跨越编译时不相关的类型。

1、所谓泛型编程就是以独立于任何特定类型的方式编写代码。使用泛型程序时,我们需要提供具体程序实例所操作的类型或值

    在泛型编程中,我们所编写的类和函数能够多态地用于跨越编译时不相关的类型

2、模板是泛型编程的基础。

3面向对象编程的多态性称为运行是多态性,应用于存在继承关系的类,我们能够编写这样的代码,忽略于基类与派生类之间的类型差异。

泛型编程所依赖的多态称为编译时多态性或参数式多态性

4、模板定义以关键字 template 开始,后接模板形参表,模板形参表是用尖括号括住的一个或多个模板形参的列表,形参之间以逗号分隔。

模板形参表不能为空

示例代码

template <typename T>
returntype functionname(parameter_list)
{
	//..
}

5、模板形参可以是表示类型的类型形参,也可以是表示常量表达式的非类型形参。非类型形参跟在类型说明符之后说明。

6使用函数模板时,编译器会推断那个(或那些)模板实参绑定到模板形参。一旦编译器确定了实际的模板实参,就称它为实例化了的函数模板的一个实例

7inline函数模板

如同非模板函数一样声明;注意inline的位置。

示例

//ok: inline specifier follows template parameter list
template <typename T> inline T min(const T&, const T&);
//error: incorrect placement of inline specifier
inline template <typename T> T min(const T&, const T&);

8、类模板

同函数模板一样,在类定义前面加template <class Type>

9、可以给模板形参赋予的唯一含义是区别形参是类型形参还是非类型形参。如果是类型形参,我们就知道该形参表示未知类型,如果是非类型形参,我们就知道它是一个未知值。

10、作用域:模板形参的名字可以在声明为模板形参之后直到模板声明或定义的末尾处使用。模板形参遵循常规名字屏蔽规则。用作模板形参的名字不能在模板内部重用,可以在不同模板中重用。

11、可以只声明,不定义模板。同一模板的声明和定义中,模板形参的名字不必相同。每个模板类型形参前面必须带上关键字typename/class,每个非类型形参前面必须带上类型名字。

示例代码

// all three uses of calc refer to the same function template
// forward declarations of the template
template <class T> T calc(const T&, const T&) ;
template <class U> U calc(const U&, const U&) ;
// actual definition of the template
template <class Type>
Type calc(const Type& a, const Type& b) { /* ... */ }

12、模板类型形参可作为类型说明符用在模板中的任何地方:返回类型,函数形参类型,变量声明,强制类型转换。

13typenameclass相同含义,而typename更直观。

14、由类型形参定义的名字可能是一个类型,也可能是一个成员值。要通过typename显示通知编译器这是一个类型。

示例

template <class Parm, class U>
Parm fcn(Parm* array, U value)
{
	Parm::size_type * p; // If Parm::size_type is a type, then a declaration
		// If Parm::size_type is an object, then multiplication
	//typename Parm::size_type *p //ok, declares p to be a pointer
}

15、非类型形参:调用函数时非类型形参将用值代替,值的类型在模板形参表中指定。

示例

// initialize elements of an array to zero
template <class T, size_t N> 
void array_init(T (&parm)[N])
{
	for (size_t i = 0; i != N; ++i) 
	{
		parm[i] = 0;
		cout << i << endl;
	}
}

int main()
{
	int x[42];
	array_init<int, 42>(x); //instantiates array_init(int(&)[42])
    //array_init(x);
	//...
	return 1;
}

对模板的非类型形参而言,求值结果相同的表达式将认为是等价的

16、编写模板代码时,对实参类型的要求尽可能少是很有益的。

17、产生模板的特定类型实例的过程称为实例化。模板在使用时将进行实例化,类模板在引用实际模板类类型时实例化,函数模板在调用它或用它对函数指针进行初始化或赋值时实例化。

18、类模板的每次实例化都会产生一个独立的类类型。类模板的特定的实例化是通过提供模板实参与每个模板形参匹配定义的。使用函数模板时,编译器通常会为我们推断模板实参。

19、从函数实参确定模板实参的类型和值的过程叫做模板实参推断。多个类型的形参与实参必须完全匹配。

20、一般而言,不会转换实参以匹配已有的实例化,会产生新的实例。除了产生新的实例化之外,编译器只会执行两种转换。

1const转换:接受const引用或const指针的函数可以分别用非const对象的引用或指针来调用,无须产生新的实例。如果函数接受非引用类型,形参类型和实参都忽略const,即无论传递const或非const对象给接受非引用类型的函数,都使用相同的实例化。

2)数组或函数到指针的转换:如果模板形参不是引用类型,则对数组或函数类型的实参应用常规指针转换。数组实参将当作指向其第一个元素的指针,函数实参当作指向函数类型的指针。

示例

template <typename T> 
T fobj(T, T); // arguments are copied
template <typename T>
T fref(const T&, const T&); // reference arguments

int main()
{

	string s1("a value");
	const string s2("another value");
	fobj(s1, s2); // ok: calls f(string, string), const is ignored
	fref(s1, s2); // ok: non const object s1 converted to const reference
	int a[10], b[42];
	fobj(a, b); // ok: calls f(int*, int*)
	fref(a, b); // error: array types don't match; arguments aren't converted to pointers
	return 1;
}

当形参为引用时,数组不能转换为指针。

21、类型转换的限制只适用于类型为模板形参的那些实参。普通类型定义的形参可以使用常规转换。

示例

template <class Type> Type sum(const Type &op1, int op2)
{
	return op1 + op2;
}

int main()
{
	double d = 3.14;
	string s1("hiya"), s2(" world");
	sum(1024, d); // ok: instantiates sum(int, int), converts d to int
	sum(1.4, d); // ok: instantiates sum(double, int), converts d to int
	sum(s1, s2); // error: s2 cannot be converted to int	
	return 1;
}

22、可以使用函数模板对函数指针进行初始化或赋值。

示例

template <typename T> 
int compare(const T&, const T&);
// pf1 points to the instantiation int compare (const int&, const int&)

int main()
{
	int (*pf1) (const int&, const int&) = compare;
	return 1;
}
目录
相关文章
|
2月前
|
存储 C++ UED
【实战指南】4步实现C++插件化编程,轻松实现功能定制与扩展
本文介绍了如何通过四步实现C++插件化编程,实现功能定制与扩展。主要内容包括引言、概述、需求分析、设计方案、详细设计、验证和总结。通过动态加载功能模块,实现软件的高度灵活性和可扩展性,支持快速定制和市场变化响应。具体步骤涉及配置文件构建、模块编译、动态库入口实现和主程序加载。验证部分展示了模块加载成功的日志和配置信息。总结中强调了插件化编程的优势及其在多个方面的应用。
337 68
|
23天前
|
安全 编译器 C++
【C++11】可变模板参数详解
本文详细介绍了C++11引入的可变模板参数,这是一种允许模板接受任意数量和类型参数的强大工具。文章从基本概念入手,讲解了可变模板参数的语法、参数包的展开方法,以及如何结合递归调用、折叠表达式等技术实现高效编程。通过具体示例,如打印任意数量参数、类型安全的`printf`替代方案等,展示了其在实际开发中的应用。最后,文章讨论了性能优化策略和常见问题,帮助读者更好地理解和使用这一高级C++特性。
38 4
|
23天前
|
算法 编译器 C++
【C++】模板详细讲解(含反向迭代器)
C++模板是泛型编程的核心,允许编写与类型无关的代码,提高代码复用性和灵活性。模板分为函数模板和类模板,支持隐式和显式实例化,以及特化(全特化和偏特化)。C++标准库广泛使用模板,如容器、迭代器、算法和函数对象等,以支持高效、灵活的编程。反向迭代器通过对正向迭代器的封装,实现了逆序遍历的功能。
33 3
|
2月前
|
安全 程序员 编译器
【实战经验】17个C++编程常见错误及其解决方案
想必不少程序员都有类似的经历:辛苦敲完项目代码,内心满是对作品品质的自信,然而当静态扫描工具登场时,却揭示出诸多隐藏的警告问题。为了让自己的编程之路更加顺畅,也为了持续精进技艺,我想借此机会汇总分享那些常被我们无意间忽视却又导致警告的编程小细节,以此作为对未来的自我警示和提升。
246 12
|
1月前
|
消息中间件 存储 安全
|
26天前
|
编译器 C++
【c++】模板详解(1)
本文介绍了C++中的模板概念,包括函数模板和类模板,强调了模板作为泛型编程基础的重要性。函数模板允许创建类型无关的函数,类模板则能根据不同的类型生成不同的类。文章通过具体示例详细解释了模板的定义、实例化及匹配原则,帮助读者理解模板机制,为学习STL打下基础。
29 0
|
2月前
|
编译器 程序员 C++
【C++打怪之路Lv7】-- 模板初阶
【C++打怪之路Lv7】-- 模板初阶
18 1
|
2月前
|
编译器 C语言 C++
C++入门6——模板(泛型编程、函数模板、类模板)
C++入门6——模板(泛型编程、函数模板、类模板)
54 0
C++入门6——模板(泛型编程、函数模板、类模板)
|
21天前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
31 2
|
27天前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
70 5