STL 函数模板

本文详细介绍了C++中的函数模板,包括其定义、使用方式和实例化过程。通过示例代码展示了如何进行类型推断以及在实际编程中的应用,同时讨论了函数模板重载和隐式类型推断的情况。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

函数模板并非真正的函数

1. 函数模板的定义

在这里插入图片描述

  • <> 表示类型参数表, 里面是类型形参
  • () 表示调用形参表,里面是调用形参
  • class 也可以换成 typename, 不过基本不用

2. 函数模板的使用

在这里插入图片描述

3. 函数模板的分析

在这里插入图片描述

#include <iostream>
using namespace std;

// 函数模板
template <class T> T Max(T x, T y){
    return x > y ? x : y;
}

// 编译器会根据 Max<int> 中的int(类型实参)从函数模板中产生一个真正的函数实体, 这个过程叫实例化
// int Max(int x, int y) { return x > y ? x : y}
// 同理 Max<double>、Max<string>
// double Max(double x, double y) { return x > y ? x : y}
// string Max(string x, string y) { return x > y ? x : y}
// 相同函数名, 相同作用域, 不同类型实参, 构成函数重载

int main(){
    int nx = 10, ny = 20;
    cout << Max<int>(nx, ny) << endl; // 除了生成函数,还会把调用的地方改成:Max(nx, ny)
    double dx = 1.1, dy = 2.2;
    cout << Max<double>(dx, dy) << endl;  // Max(dx, dy)

    return 0;
}

4. 实例化函数模板的条件

在这里插入图片描述

5. 函数模板扩展

5.1 两次编译

在这里插入图片描述
在这里插入图片描述

  • 第二次编译时,所有的未知类型都变为已知类型,会去检查这些一直类型的调用是否合理有效。
#include <iostream>
using namespace std;

class A{
public:
    void func() {
        return ;
    }
};

template <class T>void foo(){
    // 第一次编译检查:
    // xxx; 编译报错, 因为会检查函数模板内部出现的所有标识符是否有声明;
    A a;
    // 对于已知类型的调用要查看调用是否有效,比如func在A中有没有定义
    a.func();  // 已知类型调用
    // 对于未知类型都认为合理, 不管T类型中有没有定义xxxx函数
    T t;
    t.xxxx(); // 未知类型调用
    // 但是如果xxxx中有< or >, 那么就会编译报错,比如 t.x<x
}

int main(){
    // 第二次编译检查:
    // 编译报错,因为A类中没有xxxx这个成员函数
    foo<A>(); 
    return 0;
}

5.2 隐式推断类型实参

一般不认为去指定,让编译器自己干的事情,都会称为隐式。
相关可以理解为一样, 那么就可以通过x, y的类型退出T的类型。
在这里插入图片描述

#include <iostream>
using namespace std;

// 函数模板
template <class T> T Max(T x, T y){
    return x > y ? x : y;
}

int main(){
    int nx = 10, ny = 20;
    cout << Max(nx, ny) << endl; 
    // Max(nx, ny) => Max<>(nx, ny) => Max<int>(nx, ny)
    // 获得和调普通函数一致的语法表现形式
    double dx = 1.1, dy = 2.2;
    cout << Max(dx, dy) << endl; 
    return 0;
}

5.3.1 三种情况不能做隐式推断

在这里插入图片描述

  • 1中,无法根据调用形参x或y的类型推断出类型形参D的类型
  • 2中,123是int, 45.6是double,那编译器就无法推断出类型形参T的类型,因为不支持隐式的将double转为int
  • 3 看下面代码示例
#include <iostream>
using namespace std;

template <class R, class T>R Hum(T t){
    R r;
    return r;
}

int main(){
    //Hum(1); // 编译报错,返回值的类型R无法隐式推断
    Hum<double>(1); // 需要显示的指明
    return 0;
}

5.3 函数模板的重载

在这里插入图片描述

#include <iostream>
using namespace std;

template <class T> void Max(T x, T y){
    cout << "1: Max(T, T)" << endl;
}

void Max(int x, int y) {
    cout << "2: Max(int, int)" << endl;
}


int main(){
    int nx = 1, ny = 2;
    Max(nx, ny);    // 2: Max(int, int)
    double dx = 1.1, dy = 2.2; 
    Max(dx, dy);    // 1: Max(T, T)
    Max(nx, dy);    // 2: Max(int, int)
    Max<>(nx, ny);  // 1: Max(T, T)
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值