如何编写一个通用的函数?

简介: 如何编写一个通用的函数?

前言

本文主要讲解如何使用简单的模板,了解模板的原理以及基本知识.

一、函数模板

模板的作用:

C++中模板的作用是支持泛型编程。==泛型编程=是一种编程范式,它只考虑算法或数据结构的抽象,而不考虑具体的数据类型。通过使用模板,可以编写一种通用的算法或数据结构,而不需要为每种数据类型都编写一遍相关代码。模板可以用于函数、类、结构体等地方,以实现通用的算法和数据结构。使用模板可以提高代码的复用性和可读性,减少代码的重复编写。

示例:实现一个交换函数.

使用模板之前:

void swap(int& a, int& b)
{
  int tmp = a;
  a = b;
  b = tmp;
}
void swap(double& a, double& b)
{
  double tmp = a;
  a = b;
  b = tmp;
}
void swap(char& a, char& b)
{
  char tmp = a;
  a = b;
  b = tmp;
}
void test1()
{
  //交换整形
  int a = 2, b = 3;
  cout << "a=" << a << "  " << "b=" << b << endl;
  swap(a, b);
  cout << "a=" << a << "  " << "b=" << b << endl;
  //交换字符型
  char c1 = 'd', c2 = 'f';
  cout << "c1=" << c1 << "  " << "c2=" << c2 << endl;
  swap(c1, c2);
  cout << "c1=" << c1 << "  " << "c2=" << c2 << endl;
  //交换...
}

上述实现过程中使用函数重载实现.但是函数重载会有一些不合适的问题.

 1.函数重载只是重载的函数类型不同,代码复用率比较低,对于一个新的类型又要增加新的函数.

 2.由于功能基本一样,只是类型不同,导致代码的可维护性比较低,一个出错可能所有的重载均出错,均要修改.


这时,函数模板就派上用场了.

(1)函数模板的格式

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

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

其中,typename 可以使用class代替,不能使用struct代替.

示例:使用模板后的通用交换函数.

template <class T>//模板
void swap(T& a, T& b)//T会根据传参的对象进行推导为相应的类型
{
  T tmp = a;
  a = b;
  b = tmp;
}
void test1()
{
  //交换整形
  int a = 2, b = 3;
  cout << "a=" << a << "  " << "b=" << b << endl;
  swap(a, b);
  cout << "a=" << a << "  " << "b=" << b << endl;
  //交换字符型
  char c1 = 'd', c2 = 'f';
  cout << "c1=" << c1 << "  " << "c2=" << c2 << endl;
  swap(c1, c2);
  cout << "c1=" << c1 << "  " << "c2=" << c2 << endl;
  //交换...
}

(2)函数模板的原理(重点)

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

这就类似于古代的印刷术,如果每本书都需要手写,那效率是否太低了,还有各种情况可能会出错.但是印刷术的使用,就可以使用模具生成.

1c73280fc3ac4928a3d8441c7623d18b.png

函数模板的原理是通过将类型参数化,使函数能够在编译时根据实际参数的类型推断生成具体的函数实例。编译器会根据调用函数时的参数类型,实例化出适合该类型的函数版本。

比如:

当用double类型使用函数模板时,编译器通过对实参类型的推演,将T确定为double类型,然后产生一份专门处理double类型的代码.当用int类型使用函数模板时,编译器通过对实参类型的推演,将T确定为int类型,然后产生一份专门处理int类型的代码.如上图所示.

(3)模板参数的显示实例化

上面我们实现的交换函数,模板根据传参时不同的参数,自动推演出函数参数的实际类型.我们称这类通过编译器进行自动推导的实例化模板参数称为模板参数的隐式实例化.

那什么是显示实例化呢?

template <typename T>
T add(const T& a, const T& b)
{
  return a + b;
}
void test1()
{
  int a = 2, b = 3;
  double d1 = 2.5, d2 = 4.1;
  cout << add(a, b) << endl;
  cout << add(d1, d2) << endl;
  //下面这句会报错,因为一个模板参数无法在一个函数中实例化为2个不同类型的参数,一个int,一个double
  //cout << add(a, d2) << endl;
}

一个函数模板参数在同一个函数中,无法被识别为不同的两个实例类型参数,当编译器推导出a是int时,又推出d2是double类型,则编译器陷入两难.

就好比:

int:妈妈说今天不许出去玩!

double:爸爸说今天可以出去玩!

编译器:我听谁的.

解决方案:

直接将参数先强转为一样的,当模板函数接收到参数时,就只有一样的结果了.

  //解决方法1:传参时将其中不同的参数强转,使参数们相同
  cout << add(a, (int)d2) << endl;
  cout << add((double)a, d2) << endl;

模板参数的显示实例化:


让爸妈先商量好听谁的.

  //解决方法2:显示指定模板的参数
  cout << add<int>(a, d2) << endl;  //听妈妈的
  cout << add<double>(a, d2) << endl; //听爸爸的

我们应当是考虑如何在调用时采取不同的调用方式去满足我们的需求,千万不要想着去修改模板函数的返回值,参数使他们固定生成,那模板就不通用了,而且不是什么时候我们都可以去修改模板的.

错误示例:

template <typename T>
T add(const T& a, const int& b)//直接修改参数,进行固定
{
  return a + b;
}

(4)模板匹配

对于函数名相同的非模板函数和模板函数同时存在时,编译器会优先选择非模板函数.除非模板可以产生更好的匹配函数,才会选择模板.


编译器:有现成的为啥不用.

void swap(double& a, double& b)
{
  double tmp = a;
  a = b;
  b = tmp;
}
//template <class  T>
template <typename  T>//函数模板
void swap(T& a, T& b)
{
  T tmp = a;
  a = b;
  b = tmp;
}
void test1()
{
  //交换double
  double d1 = 2.5, d2 = 4.5;
  //非模板函数和模板函数同时存在时,编译器优先选择非模板函数,有现成的为啥不用?非得自己去再模具刻一个?
  swap(d1, d2);
  cout << "d1=" << d1 << "  " << "d2=" << d2 << endl;
  //交换整形
  int a = 2, b = 3;
  //没有现成的非模板函数,则编译器调用模板函数去实例化一份
  swap(a, b);
  cout << "a=" << a << "  " << "b=" << b << endl;
}

交换double型数据时,会调用void swap(double& a, double& b)函数,因为有现成的可以调用.

交换int整形时,则会调用模板函数void swap(T& a, T& b),实例化生成int型的函数.

e9699130d1654e0a9216a059fb10b820.gif

小知识:

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

因为模板函数的参数是通过参数类型进行推导的.

二、类模板

类模板的格式

template <typename T>
class A
{
  //成员
}

类模板在后续学习STL时候会具体介绍,目前了解一下即可,使用方法与函数模板类似,这里就不过多介绍了.

template <typename T>
class A
{
public:
  A(size_t capacity = 10)
    : _data(new T[capacity])
    , _size(0)
    , _capacity(capacity)
  {}
  void push_back() {
    //...
  }
  ~A()
  {
    delete _data;
    _size = 0;
    _capacity = 0;
  }
private:
  T* _data;
  size_t _size;
  size_t _capacity;
};
void test3()
{
  A<int> a1;      //实例化为存储int数据的类
  A<double> a2;   //实例化为存储double数据的类
}

本文只是对模板的初步了解,后续会遇到更加复杂的模板,比如多参数的模板等,知识一点点的学,不求速成,坚持一点点的积累,一起加油吧!

今天就讲到这里了,拜拜.



529b5f4f3453477ca75cfd3042847171.gif

目录
相关文章
|
25天前
|
开发者 Python
函数与模块:编写高效的Python代码
【4月更文挑战第8天】本文介绍了Python中提升代码效率和可读性的关键——函数和模块。函数是可重复调用的代码段,用于封装逻辑,减少重复,提高结构清晰度。通过`def`定义函数,使用`return`返回值,支持位置、关键字、默认和不定长参数。模块是包含Python代码的文件,用于组织代码,可导入使用。通过`import`导入模块,创建自定义模块以分解大型项目。熟悉Python标准库中的模块能提升开发效率。掌握函数和模块的使用对编写高效、易维护的代码至关重要。
|
8月前
|
设计模式 传感器 API
在编写RTOS代码时,如何设计一个简单、优雅、可拓展的任务初始化结构?
在编写RTOS代码时,如何设计一个简单、优雅、可拓展的任务初始化结构?
93 0
|
10月前
|
监控 数据可视化 Serverless
函数计算常用的简化配置的方式
函数计算常用的简化配置的方式
735 2
|
存储 XML 编译器
【C#基础】C# 程序通用结构
编程语言C# 程序结构的介绍 。
178 0
【C#基础】C# 程序通用结构
|
Serverless Python
函数计算中使用Python语言编写函数的运行环境信息——环境说明
函数计算中使用Python语言编写函数的运行环境信息——环境说明自制脑图
1024 0
函数计算中使用Python语言编写函数的运行环境信息——环境说明
|
存储 自然语言处理 编译器
程序环境和预处理(基本定义与实例使用)
程序环境和预处理(基本定义与实例使用)
73 0
程序环境和预处理(基本定义与实例使用)
|
数据安全/隐私保护 Python
python接口自动化(三十四)-封装与调用--函数和参数化(详解)
参数化的思维只需记住一点:不要写死,这样就便于维护,否则就会牵一发而动全身,一处修改导致处处修改,不便于维护。
1159 1
python接口自动化(三十四)-封装与调用--函数和参数化(详解)
|
安全 测试技术 API
python接口自动化(一)--什么是接口、接口优势、类型(详解)
经常听别人说接口测试,接口测试自动化,但是你对接口,有多少了解和认识,知道什么是接口吗?它是用来做什么的,测试时候要注意什么?坦白的说,笔者之前也不是很清楚。接下来先看一下接口的定义。
174 0
python接口自动化(一)--什么是接口、接口优势、类型(详解)
|
索引
Lua语言中编写模块的基本方法
Lua语言中编写模块的基本方法
130 0
|
Go C# 图形学
Unity3D中对系统类进行扩展教程(简化代码逻辑)
Unity中对系统类进行扩展的方法 Unity扩展系统类,简化代码 本文提供全流程,中文翻译。 助力快速完成 Unity 对系统类进行扩展,添加函 新建一个脚本,名称随意 类必须设为静态 Static ,函数同样(这样才能通过其他类,直接访问到扩展函数) 形参为 this + 需要扩展的类 此时,我们通过 transform.
1456 0