摘要:文章目录模板模板的概念函数模板函数模板语法函数模板注意事项函数模板案例普通函数与函数模板的区别普通函数与函数模板的调用规则模板的局限性类模板类模板语法类模板与函数模板区别类模板中成员函数创建时机类模板对象做函数参数类模
本阶段主要针对C++泛型编程和STL技术做详细讲解,探讨C++更深层的使用
模板就是建立通用的模具,大大提高复用性
模板的特点:
函数模板作用:建立一个通用函数,其函数返回值类型和形参类型可以不具体制定,用一个虚拟的类型来代表。
语法:
template<typename T>函数声明或定义
解释:
#include using namespace std;//函数模板//两个整型交换函数void swapInt(int &a, int &b) { int temp = a; a = b; b = temp;}//交换两个浮点型函数void swapDouble(double& a, double& b) { double temp = a; a = b; b = temp;}//函数模板template<typename T> //声明一个模板,告诉编译器后面代码中紧跟着的T不要报错,T是一个通用数据类型void mySwap(T& a, T& b) { T temp = a; a = b; b = temp;}void test01() { float a = 10; float b = 20; //两种方式使用函数模板 //1、自动类型推导 mySwap(a, b); cout << "a:" << a << endl; cout << "b:" << b << endl; cout << "/n"; //2、显示指定类型 mySwap<float>(a, b); cout << "a:" << a << endl; cout << "b:" << b << endl;}int main() { test01(); system("pause"); system("cls");}
总结:
注意事项:
//利用模板提供通用的交换函数template<class T>void mySwap(T& a, T& b){ T temp = a; a = b; b = temp;}// 1、自动类型推导,必须推导出一致的数据类型T,才可以使用void test01(){ int a = 10; int b = 20; char c = "c"; mySwap(a, b); // 正确,可以推导出一致的T //mySwap(a, c); // 错误,推导不出一致的T类型}// 2、模板必须要确定出T的数据类型,才可以使用template<class T>void func(){ cout << "func 调用" << endl;}void test02(){ //func(); //错误,模板不能独立使用,必须确定出T的类型 func<int>(); //利用显示指定类型的方式,给T一个类型,才可以使用该模板}int main() { test01(); test02(); system("pause"); return 0;}
总结:使用模板时必须确定出通用数据类型T,并且能够推导出一致的类型
案例描述:
利用函数模板封装一个排序的函数,可以对不同数据类型数组进行排序
排序规则从大到小,排序算法为选择排序
分别利用char数组和int数组进行测试
#include using namespace std;//实现通用 对数组进行排序的函数//从大到小 选择排序//测试 char数组、 int数组//交换函数模板template<class T>void mySwap(T& a, T& b) { T temp = a; a = b; b = temp;}//排序算法:选择排序,每次选择最大的一个放在前面template<class T>void mySort(T arr[], int len) { for (int i = 0; i < len; i++) { int max = i; //认定最大值的下标 for (int j = i + 1; j < len; j++) { if (arr[max] < arr[j]) { max = j; } } if (max != i) { //交换max和i元素 mySwap(arr[max], arr[i]); } }}//提供打印数组模板template<class T>void printArray(T arr[], int len) { for (int i = 0; i < len; i++) { cout << arr[i] << " "; } cout << endl;}void test01() { //测试char数组 char charArr[] = "badcfe"; int num = sizeof(charArr) / sizeof(char); mySort(charArr, num); printArray(charArr, num);}void test02() { //测试int数组 int intArr[] = { 7,5,1,3,9,2,4,6,8 }; int num = sizeof(intArr) / sizeof(int); mySort(intArr, num); printArray(intArr, num);}int main() { test01(); test02(); system("pause"); system("cls");}
f e d c b a9 8 7 6 5 4 3 2 1请按任意键继续. . .
普通函数与函数模板区别:
#include using namespace std;//普通函数和函数模板的区别//1、普通函数调用可以发生隐式类型转换//2、函数模板 用自动类型推导,不可以发生隐式类型转换//3、函数模板 用显示指定类型,可以发生隐式类型转换//普通函数int myAdd01(int a, int b) { return a + b;}//函数模板template<class T>T myAdd02(T a, T b) { return a + b;}void test01() { int a = 10; int b = 20; char c = "c"; //99 //cout << myAdd01(a, b) << endl; cout << myAdd01(a, c) << endl; //1、自动类型推导 不会发生隐式类型转换 //cout << myAdd02(a, c) << endl; //报错,无法推导出一致的T //2、显示指定类型 会发生隐式类型转换 cout << myAdd02<int>(a, c) << endl; cout << myAdd02<char>(a, c) << endl;}int main() { test01(); system("pause"); system("cls");}
总结:建议使用显示指定类型的方式,调用函数模板,因为可以自己确定通用类型T
调用规则如下:
#include using namespace std;//普通函数与函数模板的调用规则//1、如果函数模板和普通函数都可以调用,优先调用普通函数//2、可以通过空模板参数列表 强制调用 函数模板//3、函数模板可以发生函数重载//4、如果函数模板可以产生更好的匹配,优先调用函数模板void myPrint(int a, int b){ cout << "调用的普通函数" << endl;}template<class T>void myPrint(T a, T b) { cout << "调用的模板" << endl;}template<class T>void myPrint(T a, T b, T c) { cout << "调用重载的模板" << endl;}void test01() { int a = 10; int b = 20; myPrint(a, b); //调用的普通函数 //通过空模板参数列表,强制调用函数模板 myPrint<>(a, b); //调用模板 myPrint<>(a, b, 100); //调用重载的模板 //如果函数模板可以产生更好的匹配,优先调用函数模板 char c1 = "a"; char c2 = "b"; myPrint(c1, c2); //调用模板}int main() { test01(); system("pause"); system("cls");}
调用的普通函数调用的模板调用重载的模板调用的模板请按任意键继续. . .
总结:既然提供了函数模板,最好就不要提供普通函数,否则容易出现二义性
局限性:模板的通用性并不是万能的
template<class T> void f(T a, T b) { a = b; }
在上述代码中提供的赋值操作,如果传入的a和b是一个数组,就无法实现了
template<class T> void f(T a, T b) { if(a > b) { ... } }
在上述代码中,如果T的数据类型传入的是像Person这样的自定义数据类型,也无法正常运行
因此C++为了解决这种问题,提供模板的重载,可以为这些特定的类型提供具体化的模板
#include using namespace std;//模板局限性//特定数据类型 需要用具体方式做特殊实现class Person {public: Person(string name, int age) { this->m_Name = name; this->m_Age = age; } //姓名 string m_Name; //年龄 string m_Age;};//对比两个数据是否相等函数template<class T>bool myCompare(T& a, T& b) { if (a == b) { return true; } else { return false; }}//利用具体化Person的版本实现代码,具体化优先调用template<> bool myCompare(Person& p1, Person& p2) { if (p1.m_Name == p2.m_Name && p1.m_Age == p2.m_Age) { return true; } else { return false; }}void test01() { int a = 10; int b = 20; bool ret = myCompare(a, b); if (ret) { cout << "a == b" << endl; } else { cout << "a != b" << endl; }}void test02() { Person p1("Tom", 10); Person p2("Tom", 10); bool ret = myCompare(p1, p2); if (ret) { cout << "p1 == p2" << endl; } else { cout << "p1 != p2" << endl; }}int main() { test01(); test02(); system("pause"); system("cls");}
总结:
类模板作用:建立一个通用类,类中的成员 数据类型可以不具体制定,用一个虚拟的类型来代表。
语法:
template<typename T>类
解释:
#include using namespace std;//类模板template<class NameType, class AgeType>class Person {public: Person(NameType name, AgeType age) { this->m_Name = name; this->m_Age = age; } void showPerson() { cout << "name:" << this->m_Name << endl; cout << "age:" << this->m_Age << endl; } NameType m_Name; AgeType m_Age;};void test01() { Person<string, int> p1("张三", 18); cout << p1.m_Name << "," << p1.m_Age << endl; p1.showPerson();}int main() { test01(); system("pause"); system("cls");}
张三,18name:张三age:18请按任意键继续. . .
总结:类模板和函数模板语法相似,在声明模板template后面加类,此类称为类模板
类模板与函数模板区别主要有两点:
#include using namespace std;//类模板和函数模板区别template<class NameType, class AgeType = int>class Person {public: Person(NameType name, AgeType age) { this->m_Name = name; this->m_Age = age; } void showPerson() { cout << "name:" << this->m_Name << endl; cout << "age:" << this->m_Age << endl; } NameType m_Name; AgeType m_Age;};void test01() { //Person p("张三", 18); //1、类模板没有自动类型推导使用方式 Person<string, int> p("张三", 18); p.showPerson();}//2、类模板在模板参数列表中可以有默认参数void test02() { Person<string> p2("李四", 20); //后一个参数默认整型, 默认参数只有类模板有,函数模板没有 p2.showPerson();}int main() { test01(); test02(); system("pause"); system("cls");}
name:张三age:18name:李四age:20请按任意键继续. . .
总结:
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/122421.html
摘要:只读目的是防止程序意外地修改了它的指令。全局区存放全局变量静态变量和常量除了修饰的局部变量。程序结束时由操作系统释放。由编译器自动分配和释放。注意不要返回局部变量的地址。 ...
摘要:与都继承自类,在中也是使用字符数组保存字符串,,这两种对象都是可变的。采用字节码的好处语言通过字节码的方式,在一定程度上解决了传统解释型语言执行效率低的问题,同时又保留了解释型语言可移植的特点。 String和StringBuffer、StringBuilder的区别是什么?String为什么是不可变的? String和StringBuffer、StringBuilder的区别 可变性...
摘要:目前,中关村黑马程序员训练营已成长为行业学员质量好课程内容深企业满意的移动开发高端训练基地,并被评为中关村软件园重点扶持人才企业。黑马程序员的学员筛选制度,远比现在以上的企业招聘流程更为严格。系统的学习可以参考w3c的教程 web概念概述 * JavaWeb: * 使用Java语言开发基于互联网的项目 * 软件架构: 1. C/S: Client/Server 客户端/服务...
摘要:而面向搜索引擎,就是我们要及时的使用百度谷歌遇到问题无法解决,先别急着放弃,可以去网络寻找答案,你的坑大部分别人都已经走过了,大部分都可以找到合适的解决方案。 showImg(https://segmentfault.com/img/remote/1460000019236352?w=866&h=456); 前言: ●众多的语言,到底哪一门才是适合我的?●我们为什么要学习Java语言呢...
阅读 2439·2021-11-23 09:51
阅读 1866·2021-10-13 09:40
阅读 1372·2021-09-30 10:01
阅读 588·2021-09-26 09:46
阅读 2232·2021-09-23 11:55
阅读 1384·2021-09-10 10:51
阅读 2239·2021-09-09 09:33
阅读 2225·2019-08-29 17:25