模板

模板是泛型编程的基础,即与类型无关的逻辑代码。

利用模板机制可以显著减少冗余信息,能大幅度地节约程序代码,进一步提高面向对象程序的可重用性和可维护性。

模板是实现代码重用机制的一种工具,它可以实现类型参数化;

模板分为函数模板和类模板。

为了使用函数名相同,参数不同,返回值可同可不同的函数时,我们起初用了重载的方式。

#include<iostream>
using namespace std;int add(int a,int b)
{return a+b;
}double add(double a,double b)
{return a+b;
}int main()
{cout<<"ret1="<<add(1,2)<<endl;cout<<"ret2="<<add(2.3,3.0)<<endl;getchar();return 0;
}

wKioL1bvstuDo-moAAAPLD_E0cI751.png

但是运用重载的方式写起来比较不方便,尤其是重载的函数较多时,为了解决这类问题,我们用函数模板来实现这种功能。

#include<iostream>
using namespace std;
template <class T>
T add(const T& a,const T& b)
{cout<<"type:"<<typeid(a).name()<<endl;   //显示类型return a+b;
}int main()
{cout<<"ret1="<<add(1,2)<<endl;//还可以显示实例化,显示指定T为int,即add<int>(1,2)cout<<"ret2="<<add(2.3,3.0)<<endl;   //add<double>(2.3,3.0)getchar();return 0;
}

wKioL1bvt4bxX-QPAAARBJCMVr0319.png

以上函数中函数参数类型都是相同的,当我们遇到形如:add(1,2.3);一个参数为int,一个参数为double型,此时我们可以这样定义:

template <class T1,class T2>
T add(const T1& a,const T2& b)
{return a+b;
}

当然,肯定有人会想,模板函数可以构成重载吗?    j_0057.gif答案是肯定的。

#include<iostream>
using namespace std;int add(int a,int b)
{return a+b;
}double add(double a,double b)
{return a+b;
}
template <class T>
T add(const T& a,const T& b)
{cout<<"type:"<<typeid(a).name()<<endl; return a+b;
}int main()
{cout<<"ret1="<<add(1,2)<<endl;cout<<"ret2="<<add(2.3,3.0)<<endl;getchar();return 0;
}

当模板构成重载,调用add的函数时,它会先调用非模板类的函数,性能比较高;而模板函数内部还得调用,有推演过程判断它是什么类型,效率上会有所降低。


2.模板类

对于普通函数来说,我们拿typedef重定义一个类型,当需要改的时候,需要将int改掉就可以了;

typedef int DataType;class SEQLIST
{
private:DataType *data;
};

而我们为了适应更多的类型,于是引入了模板类,我们这样定义,体现了其高度复用的优势:

template<class T>
class SeqList
{
private:T* data;
};

写一个模板类实现SeqList的动态顺序表吧:

#include<iostream>
using namespace std;template <class T>
class SeqList
{
public:SeqList():_data(NULL),_size(0),_capacity(0){}SeqList(const SeqList<T>& s);SeqList<T>& operator=(const SeqList<T>& s);~SeqList(){if(_data != NULL){delete[] _data;}}void CheckCapacity();void PushBack(const T& d);void PopBack();void PushFront(const T& d);void PopFront();void Print();
private:T *_data;int _size;int _capacity;
};
template <class T>
SeqList<T>::SeqList(const SeqList<T>& s)
{_data = new T[s._size*sizeof(T)];int i = 0;for(i = 0;i < s._size; i++){_data[i] = s._data[i];}_size = s._size;_capacity = s._capacity;
}
template <class T>
SeqList<T>& SeqList<T>::operator=(const SeqList<T>& s)
{int i = 0;if(this == &s){return *this;}_size = s._size;_capacity = s._capacity;delete _data;_data = new T[_capacity];for(i = 0; i < _size; i++){_data[i] = s._data[i];}return *this;
}
template <class T>
void SeqList<T>::CheckCapacity()
{if(_size == _capacity){T* tmp = new T[_capacity*2+3];//memcpy(tmp,_data,_size*sizeof(T));int i = 0;for(i = 0; i < _size; i++){tmp[i] = _data[i];}delete[] _data;_data = tmp;_capacity = _capacity*2+3;}
}
template <class T>
void SeqList<T>::PushBack(const T& d)
{CheckCapacity();_data[_size] = d;_size++;
}
template <class T>
void SeqList<T>::PopBack()
{CheckCapacity();_size--;
}template <class T>	
void SeqList<T>::PushFront(const T& d)
{int i ;CheckCapacity();	for(i = _size; i > 0; i--){_data[i] = _data[i-1];}_data[0] = d;_size++;
}template <class T>
void SeqList<T>::PopFront()
{int i;CheckCapacity();for(i = 0; i < _size; i++){_data[i] = _data[i+1];}_size--;
}
template <class T>
void SeqList<T>::Print()
{int i = 0;for(i = 0; i < _size; i++){cout<<_data[i]<<" ";}cout<<endl;
}int main()
{SeqList<int> seq;SeqList<int> seq1;cout<<"seq:"<<endl;cout<<"尾插1234"<<endl;seq.PushBack(1);seq.PushBack(2);seq.PushBack(3);seq.PushBack(4);seq.Print();cout<<"尾删"<<endl;seq.PopBack();seq.Print();cout<<"头删"<<endl;seq.PopFront();seq.Print();cout<<"seq2:"<<endl;SeqList<int> seq2(seq);seq2.Print();cout<<"头插567"<<endl;seq.PushFront(5);seq.PushFront(6);seq.PushFront(7);seq.Print();seq1 = seq;cout<<"seq1:"<<endl;seq1.Print();getchar();return 0;
}

wKiom1byIpvxh_K5AAAVEIOLwCM902.png

模板就说到这里啦,有好的建议还希望大家提出来,欢迎来访哦。i_f01.gifi_f01.gif

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/382884.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C语言模拟实现标准库函数之strstr()

strstr() strstr(str1,str2) 函数用于判断字符串str2是否是str1的子串。如果是&#xff0c;则该函数返回str2在str1中首次出现的地址&#xff1b;否则&#xff0c;返回NULL。 char* my_strstr(const char* dest, const char* src) {char * str1 dest;char * str2 src;char …

linux-----强大的find

我又回来了。哈哈。今天我们来说一下linux中的另一个强大的find命令&#xff0c;灰常重要&#xff0c;灰常重要&#xff0c;灰常重要。显而易见&#xff0c;find就是对某一个文件或者目录的查找喽。但是它的一个显著的特点就是&#xff1a;一般放在后台执行&#xff0c;从整个文…

C语言模拟实现标准库函数之strchr()

strchr() 查找字符串s中首次出现字符c的位置 char * my_strchr(char *str1, char str2) {while (*str1 ! str2 && *str1 ! NULL){str1;}return str1; }int main() {char string[17];char *ptr, c r;strcpy(string, "Thisisastring");ptr my_strchr(string…

python 多人连接mysql 进行事务操作 对mysql加锁与释放锁

python 多人连接mysql 对mysql进行事务操作 对mysql加锁与释放锁 下面这个是user1代码块 # -*- coding: utf-8 -*- # user1 import pymysql import timeconn pymysql.connections.Connection(host"localhost", userdebian-sys-maint, passwordwL5wsDKDub4gT2EU…

C语言模拟实现标准库函数之strcmp()

strcmp() C/C函数&#xff0c;比较两个字符串 设这两个字符串为str1&#xff0c;str2&#xff0c; 若str1str2&#xff0c;则返回零&#xff1b; 若str1<str2&#xff0c;则返回负数&#xff1b; 若str1>str2&#xff0c;则返回正数。 char * my_strcmp(char *key,…

linux之task_struct

每个进程中都有一个进程控制块--PCB。PCB--维护进程相关的信息。然而&#xff0c;linux内核的进程控制块就是task_struct结构体&#xff0c;它可以保存进程的信息。 所有运行在系统里的进程都以task_struct链表的形式存在内核里。 每个进程都将它的信息放在task_struct结构体…

python 用元类 type 实现对数据库的ORM 映射

python 实现对数据库的ORM 映射 如果使用pymysql 操作数据库 不借助框架的话&#xff0c;频繁写sql语句, 的确比较麻烦 这里借助 type 元类 对 数据表类实现了 与mysql之间的 映射 直接上代码 import pymysqldef conn_database_execute(sql_str):conn pymysql.connect(host…

C语言模拟实现标准库函数之memcpy()

memcpy&#xff08;&#xff09; 1.如果我们需要对一个数组初始化&#xff0c;把数组的内容全部置0&#xff0c;那么能不能用strcpy() int main() {char arr1[10] { 0 };char arr2[10] " abcdefg ";strcpy(arr2, arr1);system("pause");return 0; } 我…

说说堆及堆排序

堆&#xff1a;是一种数组对象&#xff0c;它可以被看成是一种二叉树结构。 我们把堆的二叉树存储方式分为两种&#xff1a;即大堆和小堆。那么问题来了&#xff0c;什么大堆&#xff1f;什么是小堆&#xff1f; 大堆&#xff1a;让每个父节点的值都大于孩子节点的值。 小堆…

运算符优先级 速查表

运算符优先级 优先级【高到低】&#xff1a; 第一级&#xff1a; 圆括号【&#xff08;&#xff09;】、下标运算符【[]】、分量运算符的指向结构体成员运算符【->】、结构体成员运算符【.】 第二级&#xff1a; 逻辑非运算符【!】、按位取反运算符【~】、自增自减运…

linux--几种常见的进程调度算法

进程调度&#xff1a;在操作系统中调度是指一种资源分配&#xff0c;因而调度算法是指:根据系统的资源分配策略所规定的资源分配算法。操作系统管理了系统的有限资源&#xff0c;当有多个进程(或多个进程发出的请求)要使用这些资源时&#xff0c;因为资源的有限性&#xff0c;必…

指针数组和数组指针和函数指针

文章目录1.指针数组和数组指针1.int *p1[10];2.int (*p2)[10];2.函数指针char *(*fun1)(char * p1,char *p2)函数指针的概念函数指针的作用&#xff1a;例子1 .调用方式例子2&#xff1a;&#xff08;带注释&#xff09;例子33.做题的小技巧1.指针数组和数组指针 1.int *p1[10…

使用虚拟环境virtualenv 创建虚拟环境出现PermissionError: [Errno 13] Permission denied:

使用虚拟环境virtualenv 创建虚拟环境出现PermissionError: [Errno 13] Permission denied: 原因&#xff1a;虚拟环境安装的目录所属用户非当前用户 解决办法&#xff1a;将目录及其文件的所有者改为当前用户 解决命令&#xff1a;sudo chown -R 当前用户 待更改用户的目录/ …

linux之父子进程的输出

首先&#xff0c;我们来回忆一下父进程与子进程&#xff0c;前几节讲了如何创建子进程&#xff0c;像这样的&#xff0c;pid_t id fork(); 这样我们就创建好了一个子进程&#xff0c;然而fork()函数的返回值是什么呢&#xff1f;这里要记住&#xff1a;子进程返回0&#xff0c…

linux---谈谈vfork和fork的区别及exit与return

fork()&#xff1a;创建子进程的函数&#xff0c;是大家比较熟悉的吧。pid_t id fork(); 这里的vfork();也是创建子进程的函数。现在我们来剖析一下它们吧。 第一例&#xff1a; 先看一个fork()的例子哦。 对于fork()而言&#xff0c;创建子进程成功后直接打印出父子进程执…

在MySQL数据库建立多对多的数据表关系

转载自 https://blog.51cto.com/13145200724/1370753

C语言模拟实现标准库函数之qsort()

qsort 编译器函数库自带的快速排序函数。 void qsort(void*base,size_t num,size_t width,int(__cdecl*compare)(const void*,const void*)); 参数解释&#xff1a; void*base-待排序数组首地址size_t num-数组中待排序元素数量size_t width-各元素的占用空间大小int(__cde…

django contrib 包简介

转自 https://www.cnblogs.com/tianboblog/p/6955297.html

linux之管道

管道&#xff08;PIPE&#xff09;是linux中一个重要的通信方式&#xff0c;在进程中&#xff0c;我们通过从一个进程中读取到的数据转到另一个进程中的写数据中&#xff0c;这时就要有不同的进程之间共享同一份资源&#xff0c;就是所谓的进程间通信。由于进程的特点是资源独占…

把student a am i 变成 i am a student(两种方法)

文章目录#student a am i 变成 i am a student##方法1&#xff1a;指针#include <stdlib.h> #include <stdio.h> #include <string.h>void fanw(char *l, char *r) {char* left l;char* right r;char temp;while (left < right){temp *left;*left *ri…