泛型语义

  泛型(Generic Programming),即是指具有在多种数据类型上皆可操作的含意。泛型编
程的代表作品 STL 是一种高效、泛型、可交互操作的软件组件。
  泛型编程最初诞生于 C++中,目的是为了实现 C++的 STL(标准模板库)。其语言支
持机制就是模板(Templates)。
  模板的精神其实很简单:类型参数化(type parameterized),即,类型也是一种参数,
也是一种静多态。 换句话说, 把一个原本特定于某个类型的算法或类当中的类型信息抽掉,
抽出来做成模板参数。

函数模板

引子:重载函数,虽然在一定程度上达到了多类型适应的目的,但是不彻底,且有二义性存在。

#include <iostream>
using namespace std;
void myswap(int & a, int &b)
{
  int t = a;
  a = b;
  b = t;
}
void myswap(double & a, double &b)
{
  double t = a;
  a = b;
  b = t;
} int main()
{
  long a = ; long b = ;
  myswap(a,b); //ambiguous
  cout<<a<<b<<endl;
  return ;
}

函数模板

语法:在一个函数的参数表, 返回类型和函数体中使用参数化的类型。

template<typename/class 类型参数 T1, typename/class 类型参数 T2,...>

特点:结构上与普通函数无异,但是在传入参数和返回值上做了泛化

返回类型 函数模板名(函数参数列表)
{
函数模板定义体
}

案例:(概念比较抽象,请通过案例来观察其特点)

 


decltype +auto(c++11新特性)

多种类型混合模板,详情请阅读本人博文《C++11/14的新特性——更简洁》关于decltype的解析。https://www.cnblogs.com/wangkeqin/p/9285682.html

#include <iostream>

using namespace std;

template<typename R, typename T,typename U>
R add(T a, U b)
{
return a+b;
} template<typename R, typename T,typename U>
auto add2(T a, U b)->decltype(a+b)
{
return a+b;
} int main(int argc, char *argv[])
{
int a = ;
float b = 1.1;
auto ret = add<decltype(a+b),int,float>(a,b);
cout<<ret<<endl;
auto ret2 = add2<decltype(a+b)>(a,b);
cout<<ret2<<endl;
return ;
}

函数模板与普通函数的区别无非就是将函数参数高度抽象化,使其具备处理更多数据类型的能力。

特性小结

1)严格匹配,不存在隐式转化。

2)先实例化,再调用。

3)类型参数可以省略。

4)尺有所长,寸有所短。

原理:

  编译器并不是把函数模板处理成能够处理任意类的函数; 编译器从函数模板通过
具体类型产生不同的函数; 编译器会对函数模板进行两次编译: 在声明的地方对模板
代码本身进行编译, 在调用的地方对参数替换后的代码进行编译。

函数模板的应用——将快速排序算法实现模板化

#include <iostream>
#include <typeinfo>
using namespace std;

template<typename T>
void quickSort(T * array,int left, int right)
{
  if(left<right)
  {
    int low = left; int high = right;
    T pivot = array[low];
    while(low<high)
    {
      while(array[high] >= pivot && high>low )
        high--;
      array[low] = array[high];
      while(array[low] <= pivot&& high>low)
        low++;
      array[high] = array[low];
    }
  array[low] = pivot;
  quickSort(array,left,low-);
  quickSort(array,low+,right);
  }
} int main()
{
  int array[] = {,,,,,,,,,};
  quickSort<int>(array,,);
  for(auto i:array)
  {
    cout<<i<<endl;
  }
}

函数模板的默认参数

  函数模板,在调用时,先实例化为模板函数,然后再调用。当然也可以设置默认类
型的默认值。由于系统强大的自动推导能力,有时默认也没有太大的意义。

template<typename T = int>
void quickSort(T * array,int left, int right)

模板特化

  就是在实例化模板时,对特定类型的实参进行特殊处理,即实例化一个特殊的实例版本。

template<typename T> int compare( T &a, T &b)
template<> int compare < const char * >( const char* &a, const char* &b)

  当以特化定义时的形参使用模板时,将调用特化版本,模板特化分为全特化和偏特
化,函数模板的特化,只能全特化;

  比如我们在比较两个数的大小时:

#include <iostream>
#include <string.h>
using namespace std;
template<typename T> int compare( T &a, T &b)
{
  if(a > b) return ;
  else if(a < b)return -;
  else return ;
}

//实参为两个 char 指针时, 比较的是指针的大小,
//而不是指针指向内容的大小, 此时就需要为该函数模板定义一个特化版本, 即特殊处理的版本:
template<> int compare < const char * >( const char* &a, const char* &b)
{
  return strcmp(a,b);
}
int main()
{
  int a = ; int b = ;
  cout<<compare(a,b)<<endl;
  string str1 = "abc",str2 ="abc";
  cout<<compare(str1,str2)<<endl;
  char * p1 = "abc",*p2= "def";
  cout<<compare(p1,p2)<<endl;
  cout<<compare(p2,p1)<<endl;
  return ;
}

关于模板特化的认识

模板特化的原因:当前函数模板的逻辑或者功能,不能满足特定参数的需求。

模板特化的方式:将需要特化的参数提前”布置“到模板中去,提前预定模板,之后传入需要特化的参数时便可以

优先调用经过该参数特化的模板。

适用场景

  函数模板,只适用于函数的参数个数相同而类型不同,且函数体相同的情况。如果
个数不同,则不能用函数模板。

C++泛型编程之函数模板的更多相关文章

  1. 25.C++- 泛型编程之函数模板(详解)

    本章学习: 1)初探函数模板 2)深入理解函数模板 3)多参函数模板 4)重载函数和函数模板 当我们想写个Swap()交换函数时,通常这样写: void Swap(int& a, int&am ...

  2. C++STL - 函数模板

    模板主要是为了泛型编程,做到与类型无关 模板有函数模板和类模板,本文主要整理的是函数模板 1.函数模板定义 template<typename 类型形参1,typename 类型形参2,...& ...

  3. [Reprint] C++函数模板与类模板实例解析

    这篇文章主要介绍了C++函数模板与类模板,需要的朋友可以参考下   本文针对C++函数模板与类模板进行了较为详尽的实例解析,有助于帮助读者加深对C++函数模板与类模板的理解.具体内容如下: 泛型编程( ...

  4. c++函数模板---3

    原创博客:转载请标明出处:http://www.cnblogs.com/zxouxuewei/ 模板从大体上,可以分为两种:函数模板和类模板.函数模板是算法库的基础,类模板是建立标准库容器和迭代器的基 ...

  5. C++_进阶之函数模板_类模板

     C++_进阶之函数模板_类模板 第一部分 前言 c++提供了函数模板(function template.)所谓函数模板,实际上是建立一个通用函数,其函数类型和形参类型不具体制定,用一个虚拟的类型来 ...

  6. C++—模板(1)模板与函数模板

    1.引入 如何编写一个通用加法函数?第一个方法是使用函数重载, 针对每个所需相同行为的不同类型重新实现这个函数.C++的这种编程机制给编程者极大的方便,不需要为功能相似.参数不同的函数选用不同的函数名 ...

  7. C++解析(26):函数模板与类模板

    0.目录 1.函数模板 1.1 函数模板与泛型编程 1.2 多参数函数模板 1.3 函数重载遇上函数模板 2.类模板 2.1 类模板 2.2 多参数类模板与特化 2.3 特化的深度分析 3.小结 1. ...

  8. C++ 函数模板一(函数模板定义)

    //函数模板定义--数据类型做参数 #include<iostream> using namespace std; /* 函数模板声明 1.函数模板定义由模板说明和函数定义组成,并且一个模 ...

  9. Template 基础篇-函数模板(待看

    Template 基础篇-函数模板 Template所代表的泛型编程是C++语言中的重要的组成部分,我将通过几篇blog对这半年以来的学习做一个系统的总结,本文是基础篇的第一部分. Template ...

随机推荐

  1. java 对账关键点

    原理:双方交易信息对比是否平账 注意:对账bean必须重写 equals 方法 如图: //对账方法

  2. Finder 快捷键

    记录几个常用的 Finder 快捷键: 复制 Finder 里选中的路径:option+cmd+c 地址栏跳到指定路径:shift+cmd+g 增加标签:cmd+t 显示/隐藏 标签栏:shift+c ...

  3. ReboletricSample工程搭建

    受到  Just Say No to More End-to-End Tests 文章链接:http://googletesting.blogspot.tw/2015/04/just-say-no-t ...

  4. ffmpeg api升级到3.3 api变化

     void av_free_packet(AVPacket * pkt) Use void av_packet_unref(AVPacket * pkt) Wipe the packet. Unref ...

  5. apktool + eclipse 动态调试APK

    用了会AndBug,尽管挺强大的可是作为习惯了OD.EDB作为动态调试工具的人,自然有些不习惯,于是乎寻求新的动态调试解决方式.但大多数都是NetBeans + apktool.想着还得多下一个IDE ...

  6. spring定时器的配置

    首先,新建一个java项目,下面导入需要的jar包: 这里有你需要的jar包哦. jar包下载 在src文件夹下,新建一个applicationContext.xml文件 <?xml versi ...

  7. Android在有存储卡和无存储卡情况下拍照后固定尺寸和压缩大小

    我最近工作挺忙,距离上一次写博客转眼已经过了一个多月,每次学到和用到点新东西,其实都有分享的欲望,但奈何文笔太差,而一篇文章包括构思,排版,修改发布的时间最少要花费2个小时(这其中还不包括写完后未保存 ...

  8. Android项目中JNI技术生成并调用.so动态库实现详解

    生成 jni方式有两种:一种是通过SWIG从C++代码生成过度的java代码:另一种是通过javah的方式从java代码自动生成过度的C++代码.两种方式下的步骤流程正好相反. 第一种方式:由于需要配 ...

  9. UVA11551 Experienced Endeavour —— 矩阵快速幂

    题目链接:https://vjudge.net/problem/UVA-11551 题意: 给定一列数,每个数对应一个变换,变换为原先数列一些位置相加起来的和,问r次变换后的序列是多少 题解: 构造矩 ...

  10. 在Eclipse Java EE编译器中修改Web项目的发布名称

    在工程目录上右键, 选properties, 弹出属性窗口, 选中Web Project Settings, 在右边的Context root中修改保存即可 死马当做活马医 在你的工程目录下找到.se ...