C++学习笔记35:函数模板

函数模板

函数模板的目的

  • 设计通用的函数,以适应广泛的数据型式

函数模板的定义格式

  • template<模板型式参数列表>返回值型式 函数名称(参数列表);
  • 原型:template<class T> void Swap(T &a, T&b);
  • 实现:template<class T> void Swap(T &a , T&b){...}

函数模板的体化与特化

  • 针对特定型参数,在声明或第一次调用该函数模板时体化
  • 每次体化都形成针对特定型参数的重载函数版本
  • 文件最终只保留特定型参数的一份体化后的函数
  • 显式体化主要用于库的设计;显式特化覆盖体化的同型函体

//函数模板

template<class T> void f(T t) {/* */}

//显式体化:使用显式的长整型模板参数

template void f<long>(long n);

//显式体化:使用d的型式推导模板参数型式

template void f(double d);

//显式特化:使用显式的整型参数

template<>  void f<int> (int n);

//显式特化:使用c的型式推导模板参数型式

template<> void f(char c);

template <class T> void Swap(T &a, T &b)
{
    T a; t = a, a = b, b = t;
}
int main()
{
    int m = 11, n = 7;
    char a = ‘A‘, b = ‘B‘;
    double c = 1.0, d = 2.0;
    //正确调用,体化Swap(int &,int &)
    Swap(m, n);
    //正确调用,体化Swap(char &,char &)
    Swap<char>(m, n);
    //正确调用,体化Swap(double &, double &)
    Swap<double>(c, d);
    return 0;
}

函子

编写函数,求某个数据集的最小元,元素型式为T

  • 实现策略:使用函数指针作为回调函数参数
  • 实现策略:使用函子(function object, functor)作为回调函数参数

函数指针实现

//const T *a 指向数据集的基地址
//n为元素的个数
template<typename T>
const T &Min(const T *a, int n, bool(*compare)(const T&, const T&))
{
    int index = 0;
    for (int i = 1; i < n; i++)
    {
        if (comparer(a[i], a[index]))
            index = i;
    }
    return a[index];
}

函子

函子的目的

  功能上:类似函数指针

  实现上:重载函数调用操作符,必要时重载小于比较操作符

函子的优点

  函数指针不能内联,而函子可以,效率更高

  函子可以拥有任意数量的额外数据,可以保证结果和状态,提高代码的灵活性

  编译时可对函子进行型式检查

函子实现

//使用方法

int a[8] = {9,2,3,4,5,6,7,8};

int min = Min(a,8,Comparer<int>());//构造匿名函子作为函数参数

template<typename T>class Comparer
{
public:
    //确保型式T已存在或重载operator<
    bool operator()(const T &a, const T &b)
    {
        return a < b;
    }
};

template<typename T,typename Comparer>
const T &Min(const T *a, int n, Comparer comparer)
{
    int index = 0;
    for (int i = 1; i < n; i++)
    {
        if (comparer(a[i], a[index]))
            index = i;
    }
    return a[index];
}

完美转发

完美转发的意义

  • 库的设计者需要设计一个通用的函数,将接受到的参数转发给其他函数
  • 转发过程中,所有参数保持原先的语义不变

完美转发的实现策略

  • 当需要同时提供移动语义和拷贝语义时,要求重载大量建构函数,编程量大,易出错
  • 右值引用与函数模板相互配合,可以实现完美转发,极大降低代码编写量

例子:

class A
{
public:
    A(const string &s, const string &t) :_s(s), _t(t) {}
    A(const string &s, string && t) :_s(s), _t(move(t)) {}
    A(string &&s, const string &t) :_s(move(s)), _t(t) {}
    A(string &&s, string &&t) :_s(move(s)), _t(move(t)) {}
private:
    string _s, _t;
};
int main()
{
    string s1("Hello");
    const string s2("World");
    A a1(s1, s2);
    A a3(string("Good"), s2);
    A a2(s1, string("Bingo"));
    A a4(string("Good"), string("Bingo"));
    return 0;
}

改进后:

class A
{
public:
    //根据实际参数型式生成不同的左值或右值引用的建构函数版本
    //T1或T2可以不同型,此处相同仅为示例
    //实参推演时,使用引用折叠机制
    //当形式参数为T&&型时,当且仅当实际参数为右值或者右值引用时
    //实际参数型式才为右值引用
    //引用折叠机制与const/volatile无关,保持其参数性质不变
    //std::forward<T>(t)转发参数的右值引用T&&
    template<typename T1, typename T2> A(T1 &&s, T2 &&t)
        :_s(std::forward<T1>(s)), _t(std::forward<T2>(t)) {}
private:
    std::string _s, _t;
};
时间: 2025-01-06 05:15:30

C++学习笔记35:函数模板的相关文章

C++学习笔记35 方法模版

C++允许模版化类中的单个方法,这些方法可以在一个类模版中,也可以在一个非模版化的类中. 在编写一个模版化的类方法时,实际上是为不同类型编写不同版本的方法,在类模版中,方法模版对赋值运算符和复制构造函数非常有用. 要注意的是,不能用方法模版编写虚方法和析构函数. 1.一个普通类中的方法模版例子: #include <iostream> using namespace std; class man{ private: string name; public: man(const string &

C++学习笔记之函数指针

与数据项类似,函数也有地址.函数的地址是存储其机器语言代码的内存开始的地方. 一.函数指针的基础知识 假设要设计一个名为estimate()的函数,估算编写指定行数代码所需时间,并且希望不同的程序员都使用该函数,并且该函数允许每个程序员提供自己的算法来估计时间.为实现这种目标,采用的机制是,将程序员要使用的算法函数地址传给estimate(),必须完成以下工作: 获取函数地址 声明一个函数指针 用函数指针来调用函数 1.获取函数地址 使用函数名(后面不跟参数)即可.如:think()是一个函数,

C++ Primer(第五版)学习笔记_5_标准模板库string(2)

C++ Primer(第五版)学习笔记_5_标准模板库string(2) 10.搜索string对象的元素或子串 采用find()方法可查找字符串中的第一个字符元素(char, 用单引号界定)或者子串(用双引号界定):如果查到,则返回下标值(从0开始计数),如果查不到,则返回一个很大的数string:npos(即:4294967295). #include <iostream> #include <stdio.h> #include <string> using nam

C++ Primer(第五版)学习笔记_3_标准模板库vector(2)

C++ Primer(第五版)学习笔记_3_标准模板库vector(2) 欢迎大家阅读参考,如有错误或疑问请留言纠正,谢谢 6.元素的插入 insert()方法可以在vector对象的任意位置前插入一个新的元素,同时,vector自动扩张一个元素空间,插入位置后的所有元素依次向后挪动一个位置. 要注意的是,insert()方法要求插入的位置,是元素的迭代器位置,而不是元素的下标. #include <iostream> #include <vector> using namespa

C++ Primer(第五版)学习笔记_4_标准模板库string(1)

C++ Primer(第五版)学习笔记_4_标准模板库string(1) 1.创建string对象 创建一个空字符串,其长度为0 #include <iostream> #include <string> using namespace std; int main(int argc, char* argv[]) { string s; cout << s.length() << endl; return 0; } 运行结果: 0 2.给string对象赋值

C++学习笔记十六-模板和泛型编程(二)

C++学习笔记十六-模板和泛型编程(二) 16.4 类模板成员 1.模板作用域中模板类型的引用: 通常,当使用类模板的名字的时候,必须指定模板形参.这一规则有个例外:在类本身的作用域内部,可以使用类模板的非限定名.例如,在默认构造函数和复制构造函数的声明中,名字 Queue 是 Queue<Type> 缩写表示.实质上,编译器推断,当我们引用类的名字时,引用的是同一版本.因此,复制构造函数定义其实等价于: Queue<Type>(const Queue<Type> &a

C++ Primer(第五版)学习笔记_8_标准模板库_map映照容器

C++ Primer(第五版)学习笔记_8_标准模板库_map映照容器 map映照容器的元素数据是由一个键值和一个映照数据组成的,键值与映照数据之间具有一一映照的关系. map映照容器的数据结构也是采用红黑树来实现的. 1.map创建.元素插入和遍历访问 #include <iostream> #include <stdio.h> #include <vector> #include <map> #include <string> using n

MySQL学习笔记-自定义函数

MySQL学习笔记-自定义函数 1.自定义函数简介 自定义函数:用户自定义函数(user-defined function,UDF)是一种对MySQL扩展的途径,其用法与内置函数相同 自定义函数的两个必要条件:(1)参数  (2)返回值 自定义函数: 创建自定义函数 CREATE FUNCTION function_name RETURNS {STRING|INTEGER|REAL|DECIMAL} routine_body 关于函数体: 1.函数体可以由合法的SQL语句构成: 2.函数体可以是

C++ Primer 学习笔记_14_标准模板库_bitset位集合容器

C++ Primer 学习笔记_14_标准模板库_bitset位集合容器 bitset容器是一个bit位元素的序列容器,每个元素只占一个bit位,取值为0或1,因而很节省内存空间.下图是一个bitset的存储示意图,它的10个元素只使用了两个字节的空间. 使用bitset需要声明头文件"#include <bitset>" 1.创建bitset对象 创建bitset对象时,必须要指定容器的大小.bitset对象的大小一经定义,就不能修改了.下面这条语句就定义了bitset对