深入C++04:模板编程
模板编程
函数模板
模板意义:对类型也进行参数化;
函数模板:是不编译的,因为类型不知道
模板的实例化:函数调用点进行实例化,生成模板函数
模板函数:这才是要被编译器所编译的
函数模板、模板的特例化、非模板函数(普通函数):函数模板只是一个模板,模板特例化是明确类型,还有普通函数;三者不是重载关系;
模板代码是不可以在一个文件中定义,在另一个文件中使用的;模板代码调用之前,一定要看到模板定义的地方,这样的化,模板才能够进行正常的实例化,产生能够被编译器编译的代码;所以,模板代码都是放在头文件中,然后源文件当中直接进行#include包含(#include会展开代码)
类模板
利用类模板实现的顺序栈
#include<iostream>
#include<cstring>
template<typename T>
class SeqStack {
public:
SeqStack(int size = 10) //构造函数
: _pstack(new T[size])
, _top(0) //top一开始就指向空的地方,所以后面也是一直要指向空的地方
, _size(size)
{}
~SeqStack() { //析构函数
delete []_pstack;
_pstack = nullptr;
}
SeqStack<T>(const SeqStack<T> &stack) { //拷贝构造函数
_pstack = new T[stack._size];
_top = stack._top;
_size = stack._size;
for (int i = 0; i < _size; i++) {
_pstack[i] = stack._pstack[i];
}
}
SeqStack<T> operator=(const SeqStack<T> &stack) { //赋值函数重载
if (stack == *this) return *this;
delete _pstack;
T *_newstack = new T[stack._size];
for (int i = 0; i < stack._size; i++) {
_newstack[i] = stack._pstack[i];
}
_pstack = _newstack;
_size = stack._size;
_top = stack._top;
return *this;
}
void push(const T &val) {
if (full()) resize();
_pstack[_top++] = val;
}
void pop() {
if (empty()) return;
--_top;
}
T top() const {
if (empty()) throw "stack is empty!";//抛出异常也属于函数逻辑结束,不需要和返回值一致
return _pstack[_top - 1];
}
bool full() const {
return _top == _size;
}
bool empty() const {
return _top == 0;
}
private:
T *_pstack;
int _top;
int _size;
void resize() {//扩容函数
T *newstack = new T[_size * 2];
for (int i = 0; i < _size; i++) {
newstack[i] = _pstack[i];
}
_size = _size * 2;
delete []_pstack;
_pstack = newstack;
}
};
int main() {
SeqStack<int> test;
for (int i = 0; i < 20; i++) {
test.push(i);
}
for (int i = 0; i < 20; i++) {
std::cout << test.top() << std::endl;
test.pop();
}
return 0;
}
实现vector模板
#include<iostream>
template<typename T>
class vector {
public:
vector<T>(int size = 10)
: _first(new T[size])
, _last(_first)
, _end(_first + size)
{}
~vector<T>() {
delete []_first;
_first = _last = _end = nullptr;
}
vector<T>(const vector<T> &other) {
int size = other._end - other._first;
_first = new T[size];
int len = other._last - other._first;
for (int i = 0; i < len; i++) {
_first[i] = other._first[i];
}
_last = _first + len;
_end = _first + size;
}
vector<T>& operator=(const vector<T> &other) {
if (*this = other) return *this;
delete []_first;
int size = other._end - other._first;
_first = new T[size];
int len = other._last - other._first;
for (int i = 0; i < len; i++) {
_first[i] = other._first[i];
}
_last = _first + len;
_end = _first + size;
return *this;
}
void push_back(const T &val) { //向容器末尾添加元素
if (full()) resize();
*_last = val;
_last++;
}
void pop_back() {//从容器末尾删除元素
if (empty()) return;
_last--;
}
T back() const { //返回容器末尾元素
if (empty()) throw "no value";
return *(_last - 1);
}
bool full() const { return _last == _end; }
bool empty() const { return _first == _last; }
int size() const {return _last - _first;}
private:
T *_first;//数组起始位置
T *_last;//数组中有效元素的后继位置
T *_end;//数组中有效空间的后继位置
void resize() {//2倍扩容操作
int len = _end - _first;
T *newvector = new T[len * 2];
for (int i = 0; i < len; i++) {
newvector[i] = _first[i];
}
delete []_first;
_first = newvector;
_last = _first + len;
_end = _first + len * 2;
}
};
int main() {
vector<int> test;
for (int i = 0; i < 20; i++) {
test.push_back(i);
}
for (int i = 0; i < 20; i++) {
std::cout << test.back() << std::endl;
test.pop_back();
}
return 0;
}
空间配置器
上述vector模板的两个问题:
①在创建类test作为vector的对象的时候,系统默认使用了十次所存对象的构造函数(默认构造十个对象),new不仅开辟了空间,同时也生成了对象!(这样很浪费,比如用户要初始化vector一万次,实际只用一个,浪费了创建的9999个对象额外的内存和时间(内存指的是对象指向的堆内存,而存对象所需要的内存在vector里面是必须一开始就要申请的))
②创建了多少个对象就会析构多少次,上述创建那么多个对象,肯定也会析构那么多次,浪费时间;而如果把内存呢开辟和对象构造分开处理,析构的时候直接采用delete的话:只析构_first,那么只析构了这个指针所指的对象;而析构[] _first,析构会死循环,后面有内存但是不是对象;
应该:
③push_back的时候,里面原本就有对象了的,只是更改了指向,原存在的对象会丢弃,造成内存泄漏
④pop_back的时候只是_last--,没有析构对象,对象中可能还有堆空间,会找不到的;(应该要调用对象的析构函数);但是如果单纯的使用delete,不仅不会调用析构函数析构该位置的对象,还会删除该位置的内存
容器的空间配置器四件事情:内存开辟/内存释放,对象构造/对象析构;
代码:
#include<iostream>
using namespace std;
//实现自己的空间配置器
template<typename T>
class Allocator { //class
public:
T *allocate(size_t size) {//分配内存
return (T*)malloc(sizeof(T) * size);
}
void deallocate(T *p) {//释放内存
free(p);
}
void construct(T *p, const T &val) {//对象构造,明确是传入指针和引用
new (p) T(val); //定位new,指定new的区域,同时采用拷贝构造函数生成对象;
}
void destroy(T *p) {//对象析构
p->~T(); //~T()代表了T类型的析构函数;
}
};
template<typename T, typename Alloc = Allocator<T> >
class vector {
public:
vector(int size = 10) //构造函数
: _first(_allocator.allocate(size) )
, _last(_first)
, _end(_first + size)
{}
~vector() {
for (T *p = _first; p != _last; p++) {
_allocator.destroy(p);//析构有效对象
}
_allocator.deallocate(_first);//释放所有空间
_first = _last = _end = nullptr;
}
vector(const vector<T> &other) { //拷贝构造函数
int size = other._end - other._first;
// _first = new T[size];
_allocator.allcoate(size);
int len = other._last - other._first;
for (int i = 0; i < len; i++) {
// _first[i] = other._first[i];
_allocator.construct(_first + i, other._first[i]); //构造对象
}
_last = _first + len;
_end = _first + size;
}
vector& operator=(const vector<T> &other) {
if (*this = other) return *this;
// delete []_first;
//和析构的一样
for (T *p = _first; p != _last; p++) {
_allocator.destory(p);//析构有效对象
}
_allocator.deallocate(_first);//释放所有空间
//和拷贝构造的一样
int size = other._end - other._first;
int len = other._last - other._first;
_allocator.allcoate(size);
for (int i = 0; i < len; i++) {
_allocator.construct(_first + i, other._first[i]);
}
_last = _first + len;
_end = _first + size;
return *this;
}
void push_back(const T &val) { //向容器末尾添加元素
if (full()) resize();
_allocator.construct(_last, val);
_last++;
}
void pop_back() {//从容器末尾删除元素
if (empty()) return;
_allocator.destroy(--_last);
}
T back() const { //返回容器末尾元素
if (empty()) throw "no value";
return *(_last - 1);
}
bool full() const { return _last == _end; }
bool empty() const { return _first == _last; }
int size() const {return _last - _first;}
private:
T *_first;//数组起始位置
T *_last;//数组中有效元素的后继位置
T *_end;//数组中有效空间的后继位置
Alloc _allocator;//定义容器空间配置器对象!!!!!
void resize() {//2倍扩容操作
int len = _end - _first;
// T *newvector = new T[len * 2];
T *newvector = _allocator.allocate(2 * len);
for (int i = 0; i < len; i++) {
// newvector[i] = _first[i];
_allocator.construct(newvector + i, _first[i]);
}
// delete []_first;
for (T *p = _first; p != _last; p++) {
_allocator.destroy(p);
}
_allocator.deallocate(_first);
_first = newvector;
_last = _first + len;
_end = _first + len * 2;
}
};
struct Test {
Test() {
cout << "Test()" << endl;
}
~Test() {
cout << "~Test()" << endl;
}
Test(const Test &t) {
cout << "Test(const Test&)" << endl;
}
Test& operator=(const Test& t) {
cout << "operator=(const Test&)" << endl;
}
};
int main() {
Test t1;
Test t2;
vector<Test> vec;
vec.push_back(t1);
vec.push_back(t2);
cout << "===========================" << endl;
vec.pop_back();
cout << "===========================" << endl;
return 0;
}
深入C++04:模板编程的更多相关文章
- c++模板编程-typename与class关键字的区别
最近一直在研究c++模板编程,虽然有些困难,但希望能够坚持下去.今天,在书上看见一个讨论模板编程typename与class两个关键字的区别,觉得挺有意义的,就把它们给总结一下. 先看一个例子: te ...
- 实训任务04 MapReduce编程入门
实训任务04 MapReduce编程入门 1.实训1:画图mapReduce处理过程 使用有短句“A friend in need is a friend in deed”,画出使用MapReduce ...
- Hyper-v UBUNTU 12.04 模板设置
Ubuntu 12.04 模板设置 参考文档 Hyper-v安装ubuntu http://blogs.msdn.com/b/virtual_pc_guy/archive/2012/05/02/ubu ...
- C++ 11可变参数接口设计在模板编程中应用的一点点总结
概述 本人对模板编程的应用并非很深,若要用一句话总结我个人对模板编程的理解,我想说的是:模板编程是对类定义的弱化. 如何理解“类定义的弱化”? 一个完整的类有如下几部分组成: 类的名称: 类的成员变量 ...
- C++模板编程中只特化模板类的一个成员函数
模板编程中如果要特化或偏特化(局部特化)一个类模板,需要特化该类模板的所有成员函数.类模板中大多数成员函数的功能可能是一模一样的,特化时我们可能只需要重新实现1.2个成员函数即可.在这种情况下,如果全 ...
- C++之模板编程
当我们越来越多的使用C++的特性, 将越来越多的问题和事物抽象成对象时, 我们不难发现:很多对象都具有共性. 比如 数值可以增加.减少:字符串也可以增加减少. 它们的动作是相似的, 只是对象的类型不同 ...
- Django 04 模板标签(if、for、url、with、autoeacape、模板继承于引用、静态文件加载)
Django 04 模板标签(if.for.url.with.autoeacape.模板继承于引用.静态文件加载) 一.if.for.url.with.autoescape urlpatterns = ...
- c++ 基于Policy 的 模板编程
在没真正接触c++ 模板编程之前.真的没有想到c++ 还能够这么用.最大的感触是:太灵活了,太强大了. 最初接触模板威力还是在Delta3d中,感觉里面的模板使用实在是灵活与方便,特别是dtAI中使 ...
- C++模板编程中只特化模板类的一个成员函数(花样特化一个成员函数)
转自:https://www.cnblogs.com/zhoug2020/p/6581477.html 模板编程中如果要特化或偏特化(局部特化)一个类模板,需要特化该类模板的所有成员函数.类模板中大多 ...
随机推荐
- Typora 设置图片自动上传
使用 PicGo-Core(command line) 下载 PicGo-Core 依次点击 文件 -> 偏好设置 -> 图像 来到下图所示界面: 点击①位置选择 PicGo-Gore(c ...
- Java基础之浅谈异常与了解断言
一.产生错误原因 用户输入错误 设备错误 物理限制 代码错误 二.解决错误---异常 在Java中异常对象都是派生于Throwable类的一个实例. 我们一般将异常分为两种:①Error和②Excep ...
- C语言之:结构体动态分配内存(利用结构体数组保存不超过10个学生的信息,每个学生的信息包括:学号、姓名和三门课(高数、物理和英语 )的成绩和平均分(整型)。)
题目内容: 利用结构体数组保存不超过10个学生的信息,每个学生的信息包括:学号.姓名和三门课(高数.物理和英语 )的成绩和平均分(整型). 编写程序,从键盘输入学生的人数,然后依次输入每个学生的学号. ...
- JavaFX入门及相关问题
下个星期是我们专业的课程设计专周,主要是做一个Java的桌面应用程序,老师上课讲的是用Swing来开发图形化界面,但是听朋友说到一个可视化的图形界面工具JavaFX,本 人愚笨,弄了一天才大致调试完成 ...
- 关于物理机没有VMnet1和VMnet8网卡的问题
当我们在用虚拟机做实验需要与物理机进行连接时,发现无法连接上,这时候可能是没有Vmnet1或者Vmnet8网卡,又或者是Vmnet1和Vmnet8网卡都没有. 之前试过很多方法,重装.重启虚拟网络编辑 ...
- Web项目部署指南
Web项目部署指南 本文记录了部署Vue项目到阿里云服务器上的过程,其中云服务器的操作系统是CentOS 7,Web服务器用的是nginx.因为项目涉及发送异步请求,而由Flask编写的后端应用监听的 ...
- Apache Doris ODBC Mysql外表在Ubuntu下使用方法及配置
Apache Doris 社区 2022 年的总体规划,包括待开展或已开展.以及已完成但需要持续优化的功能.文档.社区建设等多方面,我们期待有更多的小伙伴参与进来讨论.同时也希望多多关注Doris,给 ...
- WinForm中TextBox文本过长解决
方案1: 如果界面有足够的空间 可以使用Multiline属性设置多行 方案2: 可以使用文本框的MouseHover事件,触发弹窗,缺点需要按确定 private void txt_Fnote_M ...
- centos下安装ansible自动化工具(超详细,包含基本使用)
ansible官网:https://www.ansible.com 众所周知,ansible是新出现的自动化运维工具,基于Python开发,集合了众多运维工具(puppet.cfengine.chef ...
- 为什么HttpContextAccessor要这么设计?
前言 周五在群里面有小伙伴问,ASP.NET Core这个HttpContextAccessor为什么改成了这个样子? 在印象中,这已经是第三次遇到有小伙伴问这个问题了,特意来写一篇记录,来回答一下这 ...