函数模版和类模版

函数模板

  • 思考:如果重载的函数,其解决问题的逻辑是一致的、函数体语句相同,只是处理的数据类型不同,那么写多个相同的函数体,是重复劳动,而且还可能因为代码的冗余造成不一致性。
  • 解决:使用模板

例:求绝对值函数的模板

 

函数模板定义语法

  • 语法形式:

    template <模板参数表>

函数定义

  • 模板参数表的内容
    • 类型参数:class(或typename) 标识符
    • 常量参数:类型说明符 标识符
    • 模板参数:template <参数表> class标识符

函数模板的示例

#include <iostream>
using namespace std;

template <class T>  //定义函数模板
void outputArray(const T *array, int count) {
    for (int i = 0; i < count; i++)
        cout << array[i] << " "; //如果数组元素是类的对象,需要该对象所属类重载了流插入运算符“<<”
    cout << endl;
}

int main() {     
    const int A_COUNT = 8, B_COUNT = 8, C_COUNT = 20;
    int a [A_COUNT] = { 1, 2, 3, 4, 5, 6, 7, 8 };
    double b[B_COUNT] = { 1.1, 2.2, 3.3, 4.4, 5.5, 6.6, 7.7, 8.8 };
    char c[C_COUNT] = "Welcome!";

    cout << " a array contains:" << endl;
    outputArray(a, A_COUNT);    
    cout << " b array contains:" << endl;
    outputArray(b, B_COUNT);    
    cout << " c array contains:" << endl;
    outputArray(c, C_COUNT);    
    return 0;
}
运行结果如下:
a array contains:
1 2 3 4 5 6 7 8
b array contains:
1.1 2.2 3.3 4.4 5.5 6.6 7.7 8.8 
c array contains:
W e l c o m e!

注意

  • 一个函数模板并非自动可以处理所有类型的数据
  • 只有能够进行函数模板中运算的类型,可以作为类型实参
  • 自定义的类,需要重载模板中的运算符,才能作为类型实参

类模板

类模板的作用

使用类模板使用户可以为类声明一种模式,使得类中的某些数据成员、某些成员函数的参数、某些成员函数的返回值,能取任意类型(包括基本类型的和用户自定义类型)。

类模板的声明

  • 类模板 template <模板参数表> class 类名 {类成员声明};
  • 如果需要在类模板以外定义其成员函数,则要采用以下的形式: template <模板参数表> 类型名 类名<模板参数标识符列表>::函数名(参数表)

类模板示例

#include <iostream>
#include <cstdlib>
using namespace std;
struct Student {
  int id;       //学号
  float gpa;    //平均分
}; 
template <class T>
class Store {//类模板:实现对任意类型数据进行存取
private:
    T item; // item用于存放任意类型的数据
    bool haveValue;  // haveValue标记item是否已被存入内容
public:
    Store();
    T &getElem();   //提取数据函数
    void putElem(const T &x);  //存入数据函数
};

template <class T>  
Store<T>::Store(): haveValue(false) { } 
template <class T>
T &Store<T>::getElem() {
    //如试图提取未初始化的数据,则终止程序
    if (!haveValue) {   
        cout << "No item present!" << endl;
        exit(1);    //使程序完全退出,返回到操作系统。
    }
    return item;        // 返回item中存放的数据 
}
template <class T>
void Store<T>::putElem(const T &x) {
    // 将haveValue 置为true,表示item中已存入数值   
    haveValue = true;   
    item = x;           // 将x值存入item
}

int main() {
    Store<int> s1, s2;  
    s1.putElem(3);  
    s2.putElem(-7);
    cout << s1.getElem() << "  " << s2.getElem() << endl;

    Student g = { 1000, 23 };
    Store<Student> s3;
    s3.putElem(g); 
    cout << "The student id is " << s3.getElem().id << endl;

    Store<double> d;
    cout << "Retrieving object D... ";
    cout << d.getElem() << endl;
   //d未初始化,执行函数D.getElement()时导致程序终止
    return 0;
}
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
函数模板和类模板是C++中非常重要的特性,它们可以大大提高代码的复用性和灵活性。在进行函数模板和类模板的实验过程中,我总结出以下几点经验: 1. 模板的定义和使用方式: 函数模板和类模板的定义方式类似,都是使用template关键字和尖括号来定义模板参数。在使用时,需要在函数名或类名后面加上尖括号,并在尖括号中指定具体的模板参数。例如: ``` template<typename T> void swap(T& a, T& b) { T temp = a; a = b; b = temp; } template<typename T> class Stack { public: void push(const T& value); T pop(); private: std::vector<T> elements; }; ``` 2. 模板参数的类型推导: 当我们调用一个模板函数或实例化一个模板类时,可以通过函数参数或构造函数参数来推导模板参数的类型。例如: ``` int a = 1, b = 2; swap(a, b); // 编译器会自动推导出T为int类型 Stack<int> intStack; intStack.push(1); // 编译器会自动推导出Stack的模板参数T为int类型 ``` 3. 模板特化: 有时候我们需要为某些特定类型的参数定义特殊的处理方式,这时可以使用模板特化。例如: ``` // 为字符串类型定义特化版本 template<> void swap<std::string>(std::string& a, std::string& b) { std::string temp = a; a = b; b = temp; } // 为bool类型定义特化版本 template<> class Stack<bool> { public: void push(bool value); bool pop(); private: std::vector<bool> elements; }; ``` 4. 模板的局限性: 模板的使用也有一些局限性,例如模板参数必须是可复制的类型,无法处理运行时动态类型,模板的实例化只能在编译期完成等。在使用模板时需要注意这些局限性。 通过这些实验,我更加深入地了解了函数模板和类模板的使用方式和特性,也学会了如何灵活地使用模板来提高代码的复用性和灵活性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值