C++泛型编程函(数模板+类模板)

目录
  • 一、函数模板
    • 1.函数模板介绍
    • 2.函数模板与重载函数的关系
    • 3.函数模板实现机制
  • 二、类模板
    • 1.类模板基本语法
    • 2.类模板内函数的整体布局【分文件使用类模板】
    • 3.类模板的static与模板类的static
    • 4.数组实现万能容器

前言:

由于C++是静态语言,也就是说使用一个数据的时候必须先指定类型,这样的操作在编译后变量的类型是无法轻易改变的,就导致扩展性太差。或者一个函数需要很多次重载的时候,代码显得冗杂,由此产生了C++函数模板。

一、函数模板

1.函数模板介绍

① 函数模板的产生背景:

  • 在编程时多多少少会因为函数参数不同写几个重载函数;
  • 函数模板的出现解决了仅仅因为参数类型不同而进行的函数重载;

解决方法:让类型作为参数传进函数或者自动类型推导,从而实现不同的功能;

② 函数模板的语法:

tmplate<typename T>

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

③ 函数模板的调用方式:

  • 1.明显的调用  函数名<参数类型>(实参);-------------常用
  • 2.自动函数推导  函数名(实参)

④ 函数模板的本质:类型参数化!

函数模板举例

  • 重载了三次的max函数,使用函数模板一次就可以解决
#include<iostream>
using namespace std;
//--------------------------------函数模板前的比较大小
int  max(int a,char b) {
    return (a > b ? a : b);
}
float max(float a, float b) {
    return (a > b ? a : b);
}
long int max(long int a, long int b) {
    return (a > b ? a : b);
}
//--------------------------------用函数模板进行比较大小
template<typename T>
T max(T& a, T& b) {
    return (a > b ? a : b);
}
int main_001() {
    int a = 10;
    int b = 20;
    char a2 = 'a',b2='b';
    cout << max<int >(a, b) << endl;
    cout<<max(a, b)<<endl;
    cout << max(a2, b2) << endl;;
    return 0;
}

2.函数模板与重载函数的关系

① 普通函数的特性:

  • 可以(隐式)进行参数类型自动转换;

② 函数模板的特性:

  • 数参数类型相同的话传进来的实参类型也必须相同(不允许自动转换);

调用规则:

  • 调用函数时优先考虑普通函数
  • 如果函数模板会有一个更好的匹配,那么选择模板函数;
  • 可以通过空模板实参列表的语法限定编译器只通过模板匹配;
  • 函数模板像普通函数一样也可以被重载

使用规则如下:

clude<iostream>
using namespace std;
//此函数模板T1 T2代表两个不同类型的参数
//所以传进来的参数也要是不同类型(可以通过简单的操作改为传相同类型的参数)
template<typename T1,typename T2>
int  myadd(T1 a, T2 b) {
    return a + b;
}
int myadd(int a, int b) {
    return a + b;
}
int myadd(int a, char b) {
    return a + b;
}
int main() {
    int a = 10;
    int b = 20;
    char c = 'c';
    cout << myadd(a,b) << endl;//----------调用add(int,int)-----优先匹配的普通函数
    cout << myadd(a,c) << endl;//----------调用add(int ,char)
    cout << myadd(c,a) << endl;//----------调用add(t1,t2)-------没有该类型的普通函数就调用模板函数
    cout << myadd(c,c) << endl;//----------调用add(t1,t2)
    cout << myadd<>(a, b) << endl;//-------强制调用add(t1,t2)
    return 0;
}

3.函数模板实现机制

① 函数模板与模板函数:

1.函数模板:------------------------------仅仅是一个模板,并未被实例化(空壳子)

template <typename T>

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

2.模板函数:------------------------------通过类型的传入,将函数模板实例化

函数模板的函数名<类型名>(参数列表);

② 函数模板机制剖析:

  • 函数模板并不会直接产生能处理任意类型的参数的函数;
  • 而是通过产生对应的模板函数实现对不同类型参数的处理;

函数模板进行两次编译:
    1.函数模板声明的地方,对函数模板代码本身进行编译
    2.将类型插入后在调用的地方对插入参数后的代码进行编译

二、类模板

1.类模板基本语法

① 单个模板类:

基本语法:

template<typename T>或template<class T>
  class 类名{private: T a;};

注意事项:

模板类是一个抽象类,定义对象时需要参数类型的传入

具体实现如下:

#include<iostream>
using namespace std;
template <class T>
class A {
public:
    void seta(T &a) {
        this->a = a;
    }
    void printA() {
        cout << this->a << endl;
    }
protected:
    T a;
};
int main() {
    int x = 888;
    A<int> a1;
    a1.seta(x);
    a1.printA();
    char xx = 'x';
    A<char> a2;
    a2.seta(xx);
    a2.printA();
    return 0;
}

② 模板类被具体类继承:

基本语法:

  定义: class 具体类名 :public 模板类名<参数类型>{};

  • 继承后的操作与普通类之间继承一样;

实现方法如下:

#include<iostream>
using namespace std;
template <class T>
class A {
public:
    void seta(T &a) {
        this->a = a;
    }
    void printA() {
        cout << this->a << endl;
    }
protected:
    T a;
};
class B :public A<int> {
private:
    int b;
public:
    void setb(int b) {
        this->b = b;
    }
    void printB() {
        cout << this->b << endl;
    }
};
int main() {
    int x = 888;
    B b1;
    b1.setb(999);
    b1.printB();
    b1.seta(x);
    b1.printA();
    return 0;
}

③ 模板类被模板类继承

类继承:

基本语法:
  template<typename T>
  class 模板类名 :public 基类模板类名<T>{ };

具体实现方法:

#include<iostream>
using namespace std;
template <class T>
class A {
public:
    void seta(T &a) {
        this->a = a;
    }
    void printA() {
        cout << this->a << endl;
    }
protected:
    T a;
};
template <class T>
class C :public A<T> {//----------语法所在地
private:
    T c;
public:
    void setC(T &c) {
        this->c = c;
    }
    void printC() {
        cout << this->c << endl;
    }
};
class B :public A<int> {
private:
    int b;
public:
    void setb(int b) {
        this->b = b;
    }
    void printB() {
        cout << this->b << endl;
    }
};
int main() {
    int p = 99;
    C<int> c1;
    c1.setC(p);
    c1.printC();
    char pp = '6';
    C<char> c2;
    c2.setC(pp);
    c2.printC();
    return 0;
}

2.类模板内函数的整体布局【分文件使用类模板】

①所有函数均在类的内部

实现方法如下:

#include<iostream>
using namespace std;
template<typename T>
class complex1 {
    friend ostream& operator<< <T>(ostream &out, complex1 &obj);
private:
    T a;
    T b;
public:
    complex1(T a=0, T b=0) {
        this->a = a;
        this->b = b;
    }
    complex1 operator+(complex1 obj) {
        complex1 tem(a+obj.a,b+obj.b);
        return tem;
    }
    void printa() {
        cout << a << endl;
    }
    void printb() {
        cout << b << endl;
    }
    
};
template<typename T>
ostream& operator<<(ostream &out, complex1<T> &obj) {
        out << obj.a << "+" << obj.b << "i" << endl;
        return out;
    }
int main_11() {
    complex1<int> a(1, 2), b(3, 4);
    complex1<int>c = a + b;
    cout << c << a << b;
    a.printa();
    a.printb();
    return 0;
}

②所有函数均在类的外部,但在同一文件

员函数实现语法:

 原型: 类名 函数名 (参数列表);

修改后的形式:  
    template <typename T>
    类名<T> 函数名 (参数列表)------参数列表该加T的就加T

流运算符 友元函数实现语法:
    原型(声明): friend 返回类型 函数名 (参数列表);

修改后的形式:
    (声明) :friend 返回类型 函数名 <T> (参数列表) ;
    template<typename T>
    (函数实现): 返回类型 函数名 (参数列表){};------类的对象做参数时修改为 类名<T>;

具体实现如下:

#include<iostream>
using namespace std;
template<typename T>
class complex2 {
    friend ostream& operator<< <T>(ostream& out, complex2& obj);
private:
    T a;
    T b;
public:
    complex2(T a = 0, T b = 0);
    complex2 operator+(complex2 obj);
    void printa();
    void printb();
};
template<typename T>
complex2<T>::complex2<T>(T a , T b ) {
    this->a = a;
    this->b = b;
}
template<typename T>
complex2<T> complex2<T>::operator+(complex2 obj) {
    complex2 tem(a + obj.a, b + obj.b);
    return tem;
}
template<typename T>
void complex2<T>::printa() {
    cout << a << endl;
}
template<typename T>
void complex2<T>::printb() {
    cout << b << endl;
}
template<typename T>
ostream& operator<<(ostream& out, complex2<T>& obj) {
    out << obj.a << "+" << obj.b << "i" << endl;
    return out;
}
int main_dd() {
    complex2<int> a(1, 2), b(3, 4);
    complex2<int>c = a + b;
    cout << c << a << b;
    a.printa();
    a.printb();
    return 0;
}

③ 所有函数均在类的外部,但在不同文件

将类分文件写后,将类函数实现的部分包含进主函数所在的文件

实现方法:

include"xxxx.cpp"

示例:

头文件:

#pragma once
#include<iostream>
using namespace std;
template<typename T>
class complex {
    friend ostream& operator<< <T>(ostream& out, complex& obj);
private:
    T a;
    T b;
public:
    complex(T a = 0, T b = 0);
    complex operator+(complex obj);
    void printa();
    void printb();
};

函数实现:

#include<iostream>
using namespace std;
#include"复数类3.h"
template<typename T>
complex<T>::complex<T>(T a, T b) {
    this->a = a;
    this->b = b;
}
template<typename T>
complex<T> complex<T>::operator+(complex obj) {
    complex tem(a + obj.a, b + obj.b);
    return tem;
}
template<typename T>
void complex<T>::printa() {
    cout << a << endl;
}
template<typename T>
void complex<T>::printb() {
    cout << b << endl;
}
template<typename T>
ostream& operator<<(ostream& out, complex<T>& obj) {
    out << obj.a << "+" << obj.b << "i" << endl;
    return out;
}

主函数:

#include<iostream>
using namespace std;
#include"复数类3h.cpp"//重点
int main() {
    complex<int> a(1, 2), b(3, 4);
    complex<int>c = a + b;
    cout << c << a << b;
    a.printa();
    a.printb();
    return 0;
}

3.类模板的static与模板类的static

类模板定义了变量,函数实现的步骤,但没有数据类型的插入,所以类模板仅仅是模板;
类模板的实现机制是程序员给出数据类型,编译器对具体的类进行实现,产生不同类型的类;
所以,类模板中的静态成员变量是某个类型的具体类独有的成员变量;只是被该类型对象所公有

区别如下:

  • 模板类中的static变量可以被该模板类的对象公用
  • 类模板的static经过类不同方式的实例化,会产生不同的static变量,
  • 且该变量只供初始化他的类使用

4.数组实现万能容器

testarray类是一个类模板,里面有一个指针类型,所以通过程序员主动实现模板类传参可以存储不同类型的数据,也就是说testarray理论可以存储任意类型的数据。

#include<iostream>
using namespace std;
class teacher {
private:
    char *name;
    char *sex;
    int age;
public:
    teacher() {
        name = NULL;
        sex = NULL;
        age = 0;
    }
    teacher(teacher& obj) {
        if (name != NULL) {
            delete [] name;
            delete[] sex;
        }
        age = obj.age;
        name = new char [sizeof(obj.name)];
        sex = new char[sizeof(obj.sex)];
        strcpy_s(name, sizeof(obj.name), obj.name);
        strcpy_s(sex, sizeof(obj.sex), sex);
    }
    void setname(char *name) {
        this->name = new char[strlen(name)+1];
        strcpy_s(this->name, strlen(name)+1, name);
    }
    void setage(int age) {
        this->age = age;
    }
    void setsex(char* sex) {
        this->sex = new char[strlen(sex)+1];
        strcpy_s(this->sex, strlen(sex)+1, sex);
    }
    friend ostream& operator<<(ostream& out, teacher& obj);
};
ostream& operator<<(ostream& out, teacher& obj) {
    cout << "姓名" << "\t" << "性别" << "\t" << "年龄" << endl;
    cout << obj.name << "\t" << obj.sex << "\t" << obj.age << endl;
    return out;
}
ostream& operator<<(ostream& out, teacher& obj);
template <typename T>
class testarray {
    friend ostream& operator<< <T>(ostream& out, testarray& obj);
private:
    int len;
    T* myarray;
public:
    testarray() {
        len = 0;
        myarray = NULL;
    }
    testarray(int len) {
        this->len=len;
        myarray = new T[len];
    }
    testarray(testarray & obj) {
        len = obj.len;
        myarray = new testarray;
        strcmp_s(myarray, len, obj.myarray);
    }
    T& operator[](int xx) {
        return myarray[xx];
    }    
};
template<typename T>
ostream& operator<<(ostream& out,testarray<T>& obj) {
    for (int i = 0;i < obj.len;i++) {
        cout << obj[i] <<" ";
    }
    cout << endl;
    return out;
}
int main() {
    testarray<int> aint(10);
    testarray<char> bchar(10);
    testarray<teacher> tea(3);
    teacher t1, t2, t3;
    char name1[]="小李",name2[]="小朱",name3[]="小黄";
    char sex1[] = "男", sex2[] = "女";
    t1.setname(name1);
    t1.setsex(sex2);
    t1.setage(40);
    t2.setname(name2);
    t2.setsex(sex1);
    t2.setage(20);
    t3.setname(name3);
    t3.setsex(sex1);
    t3.setage(28);
    tea[0] = t1;
    tea[1] = t2;
    tea[2] = t3;
    for (int i = 0;i < 10;i++) {
        aint[i] = i;
    }
    for (int i = 0;i < 10;i++) {
        bchar[i] = i + 97;
    }
    cout << aint;
    cout << bchar;
    cout << tea;
    return 0;
}

效果图:

实现思路:

  • 类模板实现对不同数据类型的变量进行处理后
  • 该变量要有针对该操作  自己处理自身的方法

换句话说就是:

  • 类模板仅仅对某种类型的处理发出指令
  • 而细枝末节的处理方式(算法),要该类型自己的方法去实现

总结:

类模板与函数模板一样也会经过两次编译,在此文中重点区分一下类模板与模板类,函数模板与模板函数的概念,泛型编程是C++开发的一大精髓,灵活地运用泛型编程对我们以后学习其他的编程语言有很大的帮助

到此这篇关于C++泛型编程函(数模板+类模板)的文章就介绍到这了,更多相关C++泛型编程函内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C++函数模板介绍

    文章转自: 公众号:Coder梁(ID:Coder_LT) 函数模板: 所谓函数的模板,本质上也就是使用泛型来定义函数. 所谓的泛型其实也就是不定的类型,比如说我们使用vector的时候,可以定义各种类型的vector,可以定义存储int型的vector也可以定义存储float类型的,也可以定义存储其他类型.我们在声明的时候将存储的类型当做参数传给了模板. 泛型可以用具体的类型,比如(int或double)替换,通过将类型作为参数传给模板,编译器会根据传递的参数类型生成该类型的函数.这种方式也被

  • 基于C++泛型编程职工管理系统

    目录 一.泛型编程思想 二.单链表是什么? 1.图示 2.链表的节点结构[节点类] 3.链表类 三.泛型编程核心 1.实现数据类 2.实现链表类 四.运行截图 1.主菜单 2.查看信息 3.更换数据类型 4.再次显示所有信息[抛转] 五.源码 前言: 前面介绍到了C++的泛型编程,并实现了万能容器,不过那使用的是数组,今天呢咱带大家实践一下使用泛型技术,结合单链表实现一个职工管理系统.保证大家看完之后有所感悟. 一.泛型编程思想 所谓泛型就是类型不固定,只需修改少量代码就可以扩展为其他类型的应用

  • C++函数模板的使用详解

    函数模板可以适用泛型来定义函数,其中泛型可以是(int, double, float)等替换.在函数重载过程中,通过将类型作为参数传递给模板,可使编译器自动产生该类型的函数. 工作原理:比如需要定义一个比大小的max函数,有三种类型的数据(int,double,float),可能就需要编写三个函数,这样既浪费时间,且容易出错.如: #include <iostream> using namespace std; int Max(int a, int b); double Max(double

  • C++泛型编程Generic Programming的使用

    目录 一.容器 array vector deque list map 键值对key/value 二.迭代器iterator(泛型指针) 三.泛型算法Generic Programming insert()插入 erase()删除 find()用于无序搜索,搜素范围[first, last), 返回iterator, 找不到则返回last copy() 复制 泛型编程最初提出时的动机很简单直接:发明一种语言机制,能够帮助实现一个通用的标准容器库. 所谓通用的标准容器库,就是要能够做到,比如用一个

  • C++中类模板的应用你了解多少

    目录 类模板应用 数组类的封装 Int的.hpp文件 int的测试文件 Person类的.hpp文件 Person类的测试文件 总结 类模板应用 数组类的封装 属性: 1,T *pAddress 指向堆区数组的指针. 2,int m_Capacity 数组容量 3,int m_Size 数组大小 行为: 1,myArray(int capacity) 构造函数 2,myArray(const MyArray&arr) 拷贝构造函数 3,operator= 重载赋值操作符= 4,operator[

  • C++模板基础之函数模板与类模板实例详解

    泛型编程  如果让你编写一个函数,用于两个数的交换.在C语言中,我们会用如下方法: // 交换两个整型 void Swapi(int* p1, int* p2) { int tmp = *p1; *p1 = *p2; *p2 = tmp; } // 交换两个双精度浮点型 void Swapd(double* p1, double* p2) { double tmp = *p1; *p1 = *p2; *p2 = tmp; }  因为C语言不支持函数重载,所以用于交换不同类型变量的函数的函数名是不

  • C++ 函数模板和类模板详情

    目录 1. 泛型编程 2. 函数模板 2.1 函数模板概念 2.2 函数模板格式化 2.3 函数模板原理 2.4 函数模板实例化 2.5 模板参数的匹配原理 3. 类模板 3.1 类模板的定义格式 3.2 类模板的实例化 1. 泛型编程 如何实现一个通用的交换函数? 在C++中可以用到函数重载 class A { public: void Swap(int& x1,int& x2) { int temp=x1; x1=x2; x2=temp; } void Swap(double&

  • C++泛型编程函(数模板+类模板)

    目录 一.函数模板 1.函数模板介绍 2.函数模板与重载函数的关系 3.函数模板实现机制 二.类模板 1.类模板基本语法 2.类模板内函数的整体布局[分文件使用类模板] 3.类模板的static与模板类的static 4.数组实现万能容器 前言: 由于C++是静态语言,也就是说使用一个数据的时候必须先指定类型,这样的操作在编译后变量的类型是无法轻易改变的,就导致扩展性太差.或者一个函数需要很多次重载的时候,代码显得冗杂,由此产生了C++函数模板. 一.函数模板 1.函数模板介绍 ① 函数模板的产

  • C++类模板与函数模板基础详细讲解

    目录 函数模板 类模板 总结 函数模板 当我们想要定义一个可以支持泛型的函数时,就要采用函数模板的方式了.所谓泛型就是可以支持多种类型的操作,比如我们定义一个compare操作,他可以根据传递给他的参数类型动态调用对应的函数版本,实现多种类型的比较. template <typename T> int compare(const T &v1, const T &v2) { if (v1 < v2) return -1; if (v2 < v1) return 1;

  • C++ 类模板与成员函数模板示例解析

    目录 类模板 类模板与成员函数模板的区别 类模板 前面以函数模板为例,介绍了具体化与实例化.那么对于类模板,有什么不同呢? 类包括成员变量和成员函数,他们都可以包含类模板的模板参数.而成员函数本身也可以是函数模板.看下面的两个类: // 类模板 template <typename T> class A { private: T t; public: void funcA(T t); }; template <typename T> void A<T>::funcA(T

  • C++函数模板与类模板实例解析

    本文针对C++函数模板与类模板进行了较为详尽的实例解析,有助于帮助读者加深对C++函数模板与类模板的理解.具体内容如下: 泛型编程(Generic Programming)是一种编程范式,通过将类型参数化来实现在同一份代码上操作多种数据类型,泛型是一般化并可重复使用的意思.泛型编程最初诞生于C++中,目的是为了实现C++的STL(标准模板库). 模板(template)是泛型编程的基础,一个模板就是一个创建类或函数的蓝图或公式.例如,当使用一个vector这样的泛型类型或者find这样的泛型函数

  • C++使用模板类实现链式栈

    本文实例为大家分享了C++使用模板类实现链式栈的具体代码,供大家参考,具体内容如下 一.实现程序: 1.Stack.h #ifndef Stack_h #define Stack_h template <class T> class Stack { public: Stack(){}; // 构造函数 void Push(const T x); // 新元素进栈 bool Pop(); // 栈顶元素出栈 virtual bool getTop(T &x) const = 0; //

  • C++ 标准模板类详解

    目录 1 标准模板库 2.泛型编程 总结 1 标准模板库 STL提供了表示容器.迭代器.函数对象和算法的模板. 容器:类似数组存储若干值,切实同质的: 迭代器:遍历容器的对象,类似遍历数组的指针,广义指针: 算法:完成特定的任务: 函数对象:类对象或函数指针. 模板类 vector erase() 删除矢量中给定区间元素.接受两个迭代器参数(该参数定义了要删除的区间),迭代器1指向区间起始处,迭代器2指向区间终止处的后一个位置. // delete first and second elemen

  • C++STL之vector模板类详解

    目录 前言 vector模板类 创建vector对象,遍历元素 迭代器 容器的基本方法 STL函数,sort 总结 前言 STL标准模板库是C++中非常有用的功能库.本篇从vector容器开始学习STL. vector模板类 创建vector对象,遍历元素 vector模板类在头文件vector中,用于存储数组,并采用动态内存分配. 创建一个vector对象并初始化长度,通过[]运算符访问元素: #include <vector> using namespace std; int main()

  • C++类模板与模板类深入详解

    1.在c++的Template中很多地方都用到了typename与class这两个关键字,而且有时候二者可以替换,那么是不是这两个关键字完全一样呢? 事实上class用于定义类,在模板引入c++后,最初定义模板的方法为:template<class T>,这里class关键字表明T是一个类型,后来为了避免class在这两个地方的使用可能给人带来混淆,所以引入了typename这个关键字,它的作用同class一样表明后面的符号为一个类型,这样在定义模板的时候就可以使用下面的方式了:      t

随机推荐