c++基础学习13-c++的函数模板
2014-09-09 10:14
330 查看
1,c++中的泛型编程
例:c++中如何交换两个变量的值?
void swap(int& a,int& b) { int t = a; a = b; b = t; } void swap(float& a,float& b) { float t = a; a = b; b = t; } void swap(char*& a,char*& b) { char* t = a; a = b; b = t; }这三个函数除了类型不同,函数体都相同,那么如何解决代码的冗余问题呢?
下面提出了泛型编程的概念:
不考虑具体数据类型的编程模式。
对于swap函数可以考虑下面的泛型写法
void swap(T& a,T& b) { T t = a; a = b; b = t; }这里面的T不是一个具体的数据类型,而是泛指任意的数据类型。
函数模板。
提供一种特殊的函数可用不同类型尽心调用。看起来和普通函数很相似,区别是类型可被参数化。
template<typename T>函数模板的语法规则
void swap(T& a,T& b) { T t = a; a = b; b = t; }
template关键字用于声明开始进行泛型编程。
typename关键字用于声明泛指类型。
函数模板的作用域:
离他最近的函数或者类。
函数模板的应用:
自动类型推倒应用。
具体类型显示调用。
#include <cstdlib> #include <iostream> using namespace std; template<typename T> void Swap(T& a, T& b) { T t = a; a = b; b = t; } void Swap (char* a, char* b) { char* t = a; a = b; b = t; } int main(int argc, char *argv[]) { int a = 1; int b = 2; Swap(a, b); cout<<"a = "<<a<<endl; cout<<"b = "<<b<<endl; float fa = 3; float fb = 4; Swap<float>(fa, fb); cout<<"fa = "<<fa<<endl; cout<<"fb = "<<fb<<endl; char ca = 'a'; char cb = 'b'; Swap<char*>(ca, cb); cout<<"ca = "<<ca<<endl; cout<<"cb = "<<cb<<endl; char cc[10] = "abc"; char cd[10] = "efg"; cout<<cc<<endl; cout<<cd<<endl; Swap(cc,cd); cout<<cc<<endl; cout<<cd<<endl; return 0; }上述第一个swap(a,b);为自动类型推导,a,b均为int所以类型参数T为int。
第二个swap<float>(fa,fb)类型显示调用用float替换参数类型T
第三个仍然是自动类型推导:ca,cb都是char,因此参数T为char。
2,深入理解函数模板
1,编译器并不是把函数模板处理成能够处理任意类型的函数。2,编译器从函数模板通过具体类型产生不同的函数。
3,编译器会对函数模板进行两次编译。
1,在声明的地方对模板代码本身进行编译。
2,在调用的地方对参数替换后的代码进行编译。
问:
当函数模板遇上函数重载会发生什么?
c++编译器优先考虑普通函数。
如果函数模板可以产生一个更好的匹配,那么选择模板。
可以通过空模板实参列表的语法,限定编译器只通过模板匹配Max<>(a,b).
#include <cstdlib> #include <iostream> using namespace std; int Max(int a, int b) { cout<<"int Max(int a, int b)"<<endl; return a > b ? a : b; } template<typename T> T Max(T a, T b) { cout<<"T Max(T a, T b)"<<endl; return a > b ? a : b; } template<typename T> T Max(T a, T b, T c) { cout<<"T Max(T a, T b, T c)"<<endl; return Max(Max(a, b), c); } int main(int argc, char *argv[]) { int a = 1; int b = 2; cout<<Max(a, b)<<endl; cout<<Max<>(a, b)<<endl; cout<<Max(3.0, 4.0)<<endl; cout<<Max(5.0, 6.0, 7.0)<<endl; cout<<Max('a', 100)<<endl; }
注意:函数模板不允许自动类型转化。
普通函数能够进行自动类型转换所以如果出现下面这段代码
cout<<Max('a',100)<<endl;那么'a'一定自动转换为int,调用普通函数。
3,多参数函数模板
1,函数模板可以定义任意多个不同的类型参数。例:
template<typename RT,typename T1,typename T2> RT more(T1 a,T2 b) { return static_cast<RT>(a+b); }调用的时候必须显示指定返回值类型,参数类型可以自动推倒,但是不能进行自动类型转换。
cout<<more<double,char,float>('a',100.0f)<<endl; cout<<more<double>('a',100.0f)<<endl;
问:
多个类型参数的模板可以进行自动类型推导吗?
当声明的类型参数为返回值类型时,无法进行自动类型推导。
4,总结
1,函数模板其实是一个具有相同行为的函数家族。
2,函数模板可以根据类型实参对函数进行推导调用。
3,函数模板可以显示指定参数的类型。
4,函数模板可以被重载。5,当为显示指定调用的函数模板时,c++编译器优先考虑普通函数。
相关文章推荐
- C++基础学习系列--2、递归思想,内置函数,函数重载,函数模板,带默认值函数
- C++ 学习之路(13):函数模板与类模板
- C++基础学习笔记 - 函数模板
- [C++再学习系列] 函数模板和类模板
- c++基础学习之函数与参数
- [C++学习历程]基础部分 C++中的函数学习
- 我的C++成长之路(-) 学习阶段 -02 函数、模板、数组的操作
- C++学习(八)函数基础
- C++语法基础--模板与泛型编程--函数模板,类模板,模板形参,非模板形参
- C++基础学习笔记----第四课(函数的重载、C和C++的相互调用)
- c++学习笔记(14.函数模板)
- c++学习---函数模板
- 【C++基础学习】关于C++静态成员函数和变量
- c++ 模板学习笔记:函数模板的类型识别(权哥)
- 【C++基础学习】关于C++静态成员函数和变量
- C++学习笔记_7:函数模板
- C++学习第15篇-函数模板
- [C++学习历程]基础部分 C++中的函数学习
- C++基础学习笔记----第四课(函数的重载、C和C++的相互调用)
- C++学习笔记(13)——利用对象、引用、指针调用虚函数