当前位置: 首页 > news >正文

常州网站建设大全沧州网站群

常州网站建设大全,沧州网站群,网页设计与网站建设课程考试,4网站建设目录 前言 1.泛型编程 2.函数模版 2.1函数模版概念 2.2函数模版格式 2.3函数模版的原理 2.4函数模版的实例化 2.5模版参数的匹配原则 3.类模版 3.1类模版的定义格式 3.2类模版的实例化 结束语 前言 前面我们学习了C的类与对象和内存管理,接下来我们继续学习…

目录

前言

1.泛型编程

2.函数模版

2.1函数模版概念

2.2函数模版格式

2.3函数模版的原理

2.4函数模版的实例化

 2.5模版参数的匹配原则

 3.类模版

3.1类模版的定义格式

3.2类模版的实例化

结束语


前言

前面我们学习了C++的类与对象和内存管理,接下来我们继续学习C++的相关知识!!!

1.泛型编程

泛型编程是一种编程范式,它允许编写与类型无关的代码,从而提高代码的重用性和灵活性。在 C++ 中,泛型编程主要通过模板实现,包括函数模板和类模板。

引入问题->如何实现一个通用的交换函数?

在以前我们都是通过写不同的类型交换函数,

void Swap(int& left, int& right)
{int temp = left;left = right;right = temp;
}
void Swap(double& left, double& right)
{double temp = left;left = right;right = temp;
}
void Swap(char& left, char& right)
{char temp = left;left = right;right = temp;
}

在这里使用函数重载虽然可以实现,但是有一下几个不好的地方:

1. 重载的函数仅仅是类型不同,代码复用率比较低,只要有新类型出现时,就需要用户自己增 加对应的函数

2. 代码的可维护性比较低,一个出错可能所有的重载均出错

那我们能否告诉编译器一个模子,让编译器根据不同的类型利用该模子来生成代码呢?

答案当然是可以的!!!

如果在C++中,也能够存在这样一个模具,通过给这个模具中填充不同材料(类型),来获得不同 材料的铸件(即生成具体类型的代码),那将会节省许多头发。巧的是前人早已将树栽好,我们只 需在此乘凉。

即就是我们所要学习的泛型编程

2.函数模版

2.1函数模版概念

函数模板代表了一个函数家族,该函数模板与类型无关,在使用时被参数化,根据实参类型产生 函数的特定类型版本。

2.2函数模版格式

template<typename T1,typename T2,...typename Tn>

返回值类型 函数名(参数列表){}

注意:typename是用来定义模板参数关键字,也可以使用class(切记:不能使用struct代替 class

例如实现交换函数

template<typename T>
void Swap(T& x, T& y) {T temp = x;x = y;y=temp;
}
int main() {double a = 3.4, b = 4.3;Swap(a, b);cout << a << " " << b << endl;char x = a, y = b;Swap(x, y);cout << x << " " << y << endl;return 0;
}

2.3函数模版的原理

函数模板是一个蓝图,它本身并不是函数,是编译器用使用方式产生特定具体类型函数的模具。 所以其实模板就是将本来应该我们做的重复的事情交给了编译器 。

 在编译器编译阶段,对于模板函数的使用,编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用。比如:当用double类型使用函数模板时,编译器通过对实参类型的推演, 将T确定为double类型,然后产生一份专门处理double类型的代码,对于字符类型也是如此。

2.4函数模版的实例化

用不同类型的参数使用函数模板时,称为函数模板的实例化。模板参数实例化分为:隐式实例化 和显式实例化。

1.隐式实例化

隐式实例化是指编译器在使用模板的函数时,根据传入的参数类型自动推导出模板参数,并生成相应的函数实例。这种方式通常在编写代码时不需要显式指定类型,编译器会根据上下文进行推导。 

template<class T>
T Add(const T& left, const T& right)
{return left + right;
}
int main()
{int a1 = 10, a2 = 20;double d1 = 10.1, d2 = 20.0;cout<<Add(a1, a2)<<endl;cout<<Add(d1, d2)<<endl;
}

如果计算a1+d1,就会出问题了,即Add(a1, d1);

/* 该语句不能通过编译,因为在编译期间,当编译器看到该实例化时,需要推演其实参类型 通过实参a1将T推演为int,通过实参d1将T推演为double类型,但模板参数列表中只有 一个T, 编译器无法确定此处到底该将T确定为int 或者 double类型而报错 */

此时有两种处理方式:1. 用户自己来强制转化 2. 使用显式实例化 3.定义多个类

Add(a, (int)d); 

Add<int>(a1, d1)

template<class T1,class T2>
T2 Add(const T1& left, const T2& right)
{
    return left + right;
}

2.显式实例化

在函数名后的<>中指定模板参数的实际类型。

int main(void){int a = 10;double b = 20.1;// 显式实例化
Add<int>(a, b);return 0;}

 2.5模版参数的匹配原则

1. 一个非模板函数可以和一个同名的函数模板同时存在,而且该函数模板还可以被实例化为这 个非模板函数。

int Add(int x, int y) {return x + y;
}
template<class T1,class T2>
T2 Add(const T1& left, const T2& right)
{return left + right;
}
int main()
{int a1 = 10, a2 = 20;double d1 = 10.1, d2 = 20.0;cout<<Add(a1, a2)<<endl; 
// 与非模板函数匹配,编译器不需要特化cout << Add<int>(a1, a2) << endl;
//调用编译器特化的Add版本
}

2.对于非模板函数和同名函数模板,如果其他条件都相同,在调动时会优先调用非模板函数而 不会从该模板产生出一个实例。如果模板可以产生一个具有更好匹配的函数, 那么将选择模板 。

// 专门处理int的加法函数
int Add(int left, int right){return left + right;
比特就业课
}// 通用加法函数
template<class T1, class T2>T1 Add(T1 left, T2 right){return left + right;}void Test(){Add(1, 2);     
// 与非函数模板类型完全匹配,不需要函数模板实例化
Add(1, 2.0);   // 模板函数可以生成更加匹配的版本,编译器根据实参生成更加匹配的
Add函数
}

3. 模板函数不允许自动类型转换,但普通函数可以进行自动类型转换。

int Add(int x, int y) {return x + y;
}
/*
template<class T1,class T2>
T2 Add(const T1& left, const T2& right)
{return left + right;
}
*/
int main()
{int a1 = 10, a2 = 20;double d1 = 10.1, d2 = 20.0;//cout<<Add(a1, a2)<<endl;//cout << Add<int>(a1, a2) << endl;cout << Add(a1, d1) << endl;//cout<<Add(d1, d2)<<endl;
}

 3.类模版

3.1类模版的定义格式

template<class T1,classT2,...class Tn>

class 类模板名

{ // 类内成员定义 };  

下面我们实现一个栈的类模版

#include <iostream>
#include <cassert>
using namespace std;
template<class T>
class Stack {
public:Stack(size_t n = 4) :_arr(new T[n]),_size(0),_capacity(n){}~Stack() {delete[] _arr;_arr = nullptr;_size = _capacity = 0;}void push(const T& x) {if (_size == _capacity) {T* temp = new T[_capacity * 2];memcpy(temp, _arr, sizeof(T) * _size);delete[]_arr;_arr = temp;_capacity = _capacity * 2;}_arr[_size++] = x;}void pop() {assert(_arr);assert(_size>0);_size--;}T top() {assert(_arr);assert(_size > 0);return _arr[_size - 1];}bool empty()const {return _size == 0;}
private:T* _arr;size_t _size;size_t _capacity;
}; 
int main() {//类模版都是显式实例化Stack <int>s1;s1.push(1);s1.push(2);s1.push(3);s1.push(4);while (!s1.empty()) {cout << s1.top()<<endl;s1.pop();}Stack <double>s2;s2.push(1.1);s2.push(2.2);s2.push(3.3);s2.push(4.4);while (!s2.empty()) {cout << s2.top() << endl ;s2.pop();}return 0;
}

相较于C语言的typedef定义数据类型(只能实现一种数据栈),这里我们可以实现多种数据类型栈。

3.2类模版的实例化

类模板实例化与函数模板实例化不同,类模板实例化需要在类模板名字后跟<>,然后将实例化的 类型放在<>中即可,类模板名字不是真正的类,而实例化的结果才是真正的类。

// Stack是类名,Stack<int>才是类型

Stack <int>s1;

Stack<double>s2;

如果我们想在类外实现Stack的构造等部分引用了类模版的函数,如果只是单纯的函数声明是仅仅不够的,我们还要再次定义一个类模版。

例如:

template<class T>
void Stack<T>:: push(const T& x) {if (_size == _capacity) {T* temp = new T[_capacity * 2];memcpy(temp, _arr, sizeof(T) * _size);delete[]_arr;_arr = temp;_capacity = _capacity * 2;}_arr[_size++] = x;
}

结束语

本期内容就到此结束,关于模版进阶的内容后续会讲解的,下节我们将对STL进行一些说明,

最后感谢各位友友的支持,点个赞吧!!!

http://www.yayakq.cn/news/958074/

相关文章:

  • 沈阳免费建网站wordpress 前端修改
  • 安徽安能建设集团网站金蝶财务软件
  • 网上商城平台运营方案太原seo排名优化公司
  • 备案的网站名与公司名称界面设计网站
  • 查一下红之易道学做的什么网站在线p图修改文字
  • 北京网站开发联系电话wordpress 明星主题
  • 主流网站开发语言有哪些浙江百度推广
  • 台州建设规划局网站wordpress the post
  • 手机自助网站建设苏州免费自助建站网站建设
  • 重庆网站推广入口深圳网站建设与推广
  • 好看的网站色彩搭配网站反链接
  • 秦皇岛网站开发报价wordpress 微语
  • 网站数据库管理系统黄山风景区
  • 高密网站建设价格卡盟做网站
  • 网站制作商wordpress评论等级
  • 网站开发前端要学什么软件易动力建设网站怎么样
  • 龙华营销型网站北京软件外包公司名单
  • 城建道桥建设集团网站苏州网络seo
  • 民治营销网站wordpress电影广告插件
  • wordpress关停网站做电商的几个网站
  • 网站优化排名网站在哪些网站做推广
  • 贵州安顺住房城乡建设网站软件开发学习路线
  • asp网站伪静态页面花钱做网站不给源码
  • 做农家乐农产品旅游的网站做线上兼职的网站
  • 网站做301怎么做网站数据统计
  • 南京建设网站首页企业信息管理系统的设计与实现
  • 网站建设佰首选金手指二七建设行业网站价格
  • 蓝色风格企业网站模板小学校园网站建设要求
  • 百度网站推广网络软件开发外包公司值不值得去
  • 建立类似淘宝的网站wordpress文件大小