STL源码解析之vector自实现
最近仔细阅读了下侯捷老师的《STL源码剖析》的前三章内容:空间配置器、迭代器、traits技术,为了加强理解和掌握,特参照源码自实现Vector,以下对相关重点知识进行说明。
1. vector实现框架
2. 空间配置器
空间配置器方面的内容在之前的博客已进行详细说明,查看->STL空间配置器解析和实现.
3. 内存基本处理工具
(1)对象构造
(2)Destroy()函数的泛化和特化版本实现,主要体现对象T是否包含trivial construct
(3)迭代器范围赋值
(4)迭代器范围拷贝
template<class T1, class T2>
void Construct(T1* p, const T2& value)
{
new (p) T1(value); // placement new;调用T1::T1(value)
} template<class T>
void Destroy(T* p)
{
p->~T();
p = NULL;
} template<class ForwardIterator>
void Destroy(ForwardIterator first, ForwardIterator last)
{
ForwardIterator iter = first;
while (iter != last)
{
typedef typename TRAITS_DEFINE::iterator_traits<ForwardIterator>::value_type T;
cout << typeid(T).name();
T p = (T)*iter;
Destroy(&p);
iter++;
}
} template<class ForwardIterator>
void Destroy(ForwardIterator first, ForwardIterator last, TRAITS_DEFINE::__true_type)
{
;
} template<class ForwardIterator>
void Destroy(ForwardIterator first, ForwardIterator last, TRAITS_DEFINE::__false_type)
{
for (; first<last; first++)
{
Destroy(&*first)
}
} template<class ForwardIterator, class T>
void Destroy(ForwardIterator first, ForwardIterator last, T*)
{
typedef typename TRAITS_DEFINE::__type_traits<T>::has_trivial_destructor trivial_destructor;
Destroy(first, last, trivial_destructor());
}
template<class OutputIterator>
OutputIterator uninitialized_fill_n_ex(OutputIterator iter, const size_type& nSize, const T& nValue)
{
for (size_t i = ; i < nSize; i++)
{
*iter++ = nValue;
} return iter;
} void Fill_Initialize(const size_type& nSize, const T& nValue)
{
Iterator iter = data_allocator::Allocate(nSize);
uninitialized_fill_n_ex(iter, nSize, nValue); Start = iter;
Finish = Start + nSize;
EndOfStorage = Finish;
} template<class InputIterator, class OutputIterator>
OutputIterator uninitialzed_copy_ex(InputIterator first, InputIterator last, OutputIterator dest)
{
for (; first != last; first++,dest++)
{
*dest = *first;
} return dest;
}
4. iterator_traites和type_traits
类型萃取实现的关键计数在于“模版特例化”,通过class template partial specialization的作用,不论是原生指针或class-type iterators,都可以让外界方便地取其相应类别
原生指针不是class type,如果不是class type就无法为它定义内嵌型别。但模板特例化解决了该问题
template<class T>
class C{...}; // 这个泛化版本允许(接受)T为任何型别
template<class T>
class C<T*>{...}; // 这个特化版本仅适用于“T为原生指针”的情况
type_traits类型萃取,判断T类型中是否含有trivial construct,这样在destroy函数的就可以根据其做相应的处理
namespace TRAITS_DEFINE
{
template<class T>
struct iterator_traits
{
typedef T value_type;
}; template<class T>
struct iterator_traits<T *>
{
typedef T value_type;
}; struct __true_type{};
struct __false_type{};
template<class type>
struct __type_traits
{
typedef __true_type this_dummy_must_be_first;
typedef __false_type has_trivial_default_constructor;
typedef __false_type has_trivial_copy_constructor;
typedef __false_type has_trivial_assignment_operator;
typedef __false_type has_trivial_destructor;
typedef __false_type is_POD_type;
}; template<>
struct __type_traits<int>
{
typedef __true_type this_dummy_must_be_first;
typedef __true_type has_trivial_default_constructor;
typedef __true_type has_trivial_copy_constructor;
typedef __true_type has_trivial_assignment_operator;
typedef __true_type has_trivial_destructor;
typedef __true_type is_POD_type;
};
}
5. vector代码实现
5.1. 构造
typedef T& Reference;
typedef T* Iterator;
typedef T value_type;
typedef size_t size_type;
typedef CSimpleAlloc<value_type> data_allocator; // 自定义空间配置器 Iterator Start;
Iterator Finish;
Iterator EndOfStorage;
Vector():Start(NULL), Finish(NULL),EndOfStorage(NULL){
cout << "Vector()" << endl;
}
Vector(const size_type& nSize, const T& nValue)
{
Fill_Initialize(nSize, nValue);
}
explicit Vector(const size_type& nSize)
{
Fill_Initialize(nSize, T());
}
void Fill_Initialize(const size_type& nSize, const T& nValue)
{
Iterator iter = data_allocator::Allocate(nSize);
uninitialized_fill_n_ex(iter, nSize, nValue); Start = iter;
Finish = Start + nSize;
EndOfStorage = Finish;
}
5.2. 析构
void Deallocate()
{
if (Start)
{
data_allocator::Deallocate(Start, EndOfStorage-Start);
}
}
~Vector()
{
Destroy(Start, Finish);
Deallocate();
}
5.3. Push
void Push(const T& Value)
{
if (Finish != EndOfStorage)
{
Construct(Finish, Value); // 全局函数
++Finish;
}
else
{
// 容器已满,需扩容
Insert_Aux(End(), Value);
}
}
void Insert_Aux(Iterator position, const T& value)
{
if (Finish != EndOfStorage)
{
// 还有备用空间可用
Construct(Finish, *(Finish - ));
++Finish;
T x_copy = value;
//copy_backward(position, Finish - 2, Finish - 1);
*position = x_copy;
}
else
{
// 已无备用空间
size_type nOldSize = Size();
size_type nNewSize = (nOldSize == ) ? : * nOldSize;
Iterator NewStart = data_allocator::Allocate(nNewSize);
Iterator NewFinish = NewStart;
try
{
NewFinish = uninitialzed_copy_ex(Start, position, NewStart);
Construct(NewFinish, value);
NewFinish++;
}
catch (...)
{
Destroy(NewStart, NewFinish);
data_allocator::Deallocate(NewStart, nNewSize);
throw;
} Destroy(Begin(),End());
data_allocator::Deallocate(Start, EndOfStorage - Start); Start = NewStart;
Finish = NewFinish;
EndOfStorage = Start + nNewSize;
}
}
5.4. Pop_Back
void Pop_Back()
{
if (!Empty())
{
--Finish;
Destroy(Finish);
}
}
5.5. Erase
Erase方法只能清空元素,但不能释放空间
Iterator Erase(Iterator pos)
{
if (pos + != Finish)
{
copy(pos + , Finish, pos);
}
--Finish;
destroy(Finish);
return pos;
}
Iterator Erase(Iterator first, Iterator last)
{
Iterator iter = uninitialzed_copy_ex(last, Finish, first);
Destroy(iter, Finish);
Finish = Finish - (last - first);
return first;
}
5.6. Resize
void Resize(size_type new_size, const T& x) {
//如果新空间大小 小于size 就裁去多余的 如果新空间大于size那么就将这些插入.
if (new_size < Size())
Erase(Begin() + new_size, End());
else
{
Iterator NewStart = data_allocator::Allocate(new_size);
Iterator NewFinish = NewStart;
try
{
NewFinish = uninitialzed_copy_ex(Start, Finish, NewStart);
for (size_t i = ; i < (new_size-Size()); i++)
{
*NewFinish++ = x; }
}
catch (...)
{
Destroy(NewStart, NewFinish);
data_allocator::Deallocate(NewStart, new_size);
throw;
} Destroy(Begin(), End());
data_allocator::Deallocate(Start, EndOfStorage - Start); Start = NewStart;
Finish = NewFinish;
EndOfStorage = Start + new_size;
}
}
5.7. 全部实现
#pragma once
#include "simple_alloc.h"
#include <iostream>
#include<memory>
using namespace std; /*
Construct和Destroy实现
*/
namespace TRAITS_DEFINE
{
template<class T>
struct iterator_traits
{
typedef T value_type;
}; template<class T>
struct iterator_traits<T *>
{
typedef T value_type;
}; struct __true_type{};
struct __false_type{};
template<class type>
struct __type_traits
{
typedef __true_type this_dummy_must_be_first;
typedef __false_type has_trivial_default_constructor;
typedef __false_type has_trivial_copy_constructor;
typedef __false_type has_trivial_assignment_operator;
typedef __false_type has_trivial_destructor;
typedef __false_type is_POD_type;
}; template<>
struct __type_traits<int>
{
typedef __true_type this_dummy_must_be_first;
typedef __true_type has_trivial_default_constructor;
typedef __true_type has_trivial_copy_constructor;
typedef __true_type has_trivial_assignment_operator;
typedef __true_type has_trivial_destructor;
typedef __true_type is_POD_type;
};
} // 内存管理工具
template<class T1, class T2>
void Construct(T1* p, const T2& value)
{
new (p) T1(value); // placement new;调用T1::T1(value)
} template<class T>
void Destroy(T* p)
{
p->~T();
p = NULL;
} template<class ForwardIterator>
void Destroy(ForwardIterator first, ForwardIterator last)
{
ForwardIterator iter = first;
while (iter != last)
{
typedef typename TRAITS_DEFINE::iterator_traits<ForwardIterator>::value_type T;
cout << typeid(T).name();
T p = (T)*iter;
Destroy(&p);
iter++;
}
} template<class ForwardIterator>
void Destroy(ForwardIterator first, ForwardIterator last, TRAITS_DEFINE::__true_type)
{
;
} template<class ForwardIterator>
void Destroy(ForwardIterator first, ForwardIterator last, TRAITS_DEFINE::__false_type)
{
for (; first<last; first++)
{
Destroy(&*first)
}
} template<class ForwardIterator, class T>
void Destroy(ForwardIterator first, ForwardIterator last, T*)
{
typedef typename TRAITS_DEFINE::__type_traits<T>::has_trivial_destructor trivial_destructor;
Destroy(first, last, trivial_destructor());
} /*
* 实现vector
* vector基本操作:构造、resize、erase、push、pop
*/
template <typename T>
class Vector
{
public:
typedef T& Reference;
typedef T* Iterator;
typedef T value_type;
typedef size_t size_type;
typedef CSimpleAlloc<value_type> data_allocator; // 自定义空间配置器 Iterator Start;
Iterator Finish;
Iterator EndOfStorage; public:
Vector():Start(NULL), Finish(NULL),EndOfStorage(NULL){
cout << "Vector()" << endl;
}
Vector(const size_type& nSize, const T& nValue)
{
Fill_Initialize(nSize, nValue);
}
explicit Vector(const size_type& nSize)
{
Fill_Initialize(nSize, T());
}
~Vector()
{
Destroy(Start, Finish);
Deallocate();
}
Iterator Begin(){return Start;}
Iterator End(){return Finish;}
size_type Size() const { return size_type(Finish - Start); }
size_type Capacity() const { return size_type(EndOfStorage - Start); }
bool Empty() { return Begin() == End(); }
Reference operator[](const size_type& nIndex)
{
return *(start + n);
} Reference Front(){ return *Start; }
Reference Back(){ return *(--Finish); } void Push(const T& Value)
{
if (Finish != EndOfStorage)
{
Construct(Finish, Value); // 全局函数
++Finish;
}
else
{
// 容器已满,需扩容
Insert_Aux(End(), Value);
} }
void Pop_Back()
{
if (!Empty())
{
--Finish;
Destroy(Finish);
}
} void Print()
{
cout << "打印vector所有元素: ";
Iterator iter = Start;
while (iter != Finish)
{
cout << *iter << " ";
iter++;
}
cout << endl;
} Iterator Erase(Iterator pos)
{
if (pos + != Finish)
{
copy(pos + , Finish, pos);
}
--Finish;
destroy(Finish);
return pos;
}
Iterator Erase(Iterator first, Iterator last)
{
Iterator iter = uninitialzed_copy_ex(last, Finish, first);
Destroy(iter, Finish);
Finish = Finish - (last - first);
return first;
} void Resize(size_type new_size, const T& x) {
//如果新空间大小 小于size 就裁去多余的 如果新空间大于size那么就将这些插入.
if (new_size < Size())
Erase(Begin() + new_size, End());
else
{
Iterator NewStart = data_allocator::Allocate(new_size);
Iterator NewFinish = NewStart;
try
{
NewFinish = uninitialzed_copy_ex(Start, Finish, NewStart);
for (size_t i = ; i < (new_size-Size()); i++)
{
*NewFinish++ = x; }
}
catch (...)
{
Destroy(NewStart, NewFinish);
data_allocator::Deallocate(NewStart, new_size);
throw;
} Destroy(Begin(), End());
data_allocator::Deallocate(Start, EndOfStorage - Start); Start = NewStart;
Finish = NewFinish;
EndOfStorage = Start + new_size;
}
} void Clear()
{
Erase(Start, Finish);
}
private: void Deallocate()
{
if (Start)
{
data_allocator::Deallocate(Start, EndOfStorage-Start);
}
}
void Insert_Aux(Iterator position, const T& value)
{
if (Finish != EndOfStorage)
{
// 还有备用空间可用
Construct(Finish, *(Finish - ));
++Finish;
T x_copy = value;
//copy_backward(position, Finish - 2, Finish - 1);
*position = x_copy;
}
else
{
// 已无备用空间
size_type nOldSize = Size();
size_type nNewSize = (nOldSize == ) ? : * nOldSize;
Iterator NewStart = data_allocator::Allocate(nNewSize);
Iterator NewFinish = NewStart;
try
{
NewFinish = uninitialzed_copy_ex(Start, position, NewStart);
Construct(NewFinish, value);
NewFinish++;
}
catch (...)
{
Destroy(NewStart, NewFinish);
data_allocator::Deallocate(NewStart, nNewSize);
throw;
} Destroy(Begin(),End());
data_allocator::Deallocate(Start, EndOfStorage - Start); Start = NewStart;
Finish = NewFinish;
EndOfStorage = Start + nNewSize;
}
} // 申请并填充内存
template<class OutputIterator>
OutputIterator uninitialized_fill_n_ex(OutputIterator iter, const size_type& nSize, const T& nValue)
{
for (size_t i = ; i < nSize; i++)
{
*iter++ = nValue;
} return iter;
} void Fill_Initialize(const size_type& nSize, const T& nValue)
{
Iterator iter = data_allocator::Allocate(nSize);
uninitialized_fill_n_ex(iter, nSize, nValue); Start = iter;
Finish = Start + nSize;
EndOfStorage = Finish;
} template<class InputIterator, class OutputIterator>
OutputIterator uninitialzed_copy_ex(InputIterator first, InputIterator last, OutputIterator dest)
{
for (; first != last; first++,dest++)
{
*dest = *first;
} return dest;
}
};
vector_define.h
// 完全解析STL空间配置器
/***** 一级配置区 ****************************/
// 1. 采用mallo/relloc/free申请和释放内存
// 2. 处理内存申请失败的处理
// (1)设置set_new_handle,若为NULL抛出__THROW_BAD_ALLOC;
// (2)尝试重复申请
/**********************************************/
#pragma once class CFirstLevelAlloc;
class CSecondLevelAlloc; #ifdef _CHUNK_ALLOC
typedef CFirstLevelAlloc SelfAlloc;
#else
typedef CSecondLevelAlloc SelfAlloc;
#endif typedef void(*CallBackFunc)();
class CFirstLevelAlloc
{
public:
CFirstLevelAlloc(); static CallBackFunc m_CallBackFunc;
static void* Allocate(size_t nSize);
static void* Allocate(void *p, size_t nSize);
static void Deallocate(void *p, size_t nSize = );
static void SetCallBackHandle(CallBackFunc cb); private:
static void* ReAllocate(size_t nSize);
static void* ReAllocate(void *p, size_t nSize);
}; enum {ALIGN = }; // 小型区块的上调边界
enum {MAX_BYTES = }; // 小型区块的上限
enum {FREELISTNUM = MAX_BYTES/ALIGN}; // free-lists个数 // 空闲内存链表结构
union FreeList
{
union FreeList *pFreeList;
char client_data[];
}; class CSecondLevelAlloc
{
public:
CSecondLevelAlloc();
static void* Allocate(size_t nSize);
static void Deallocate(void *p, size_t nSize);
static void SetCallBackHandle(CallBackFunc cb); private:
static size_t FreeListIndex(int nBytes); // 根据区块大小得到freelist索引
static size_t Round_Up(int nBytes); // 将bytes按内存对齐上调至ALIGN的倍数
static char *ChunkAlloc(size_t nSize, int& nObjs);
static void* Refill(size_t nSize); private:
static FreeList *m_pFreeList[FREELISTNUM];
static char *m_pStartFree;
static char *m_pEndFree;
static size_t m_nHeapSize;
};
alloc_define.h
#pragma once #include "alloc_define.h"
template<typename T, typename Alloc = SelfAlloc>
class CSimpleAlloc
{
public:
static T* Allocate(size_t n)
{
if (n == )
{
return NULL;
}
return (T *)Alloc::Allocate(n * sizeof(T));
} static T* Allocate(void)
{
return (T *)Alloc::Allocate(sizeof(T));
} static void Deallocate(T *p, size_t n)
{
if (n != )
{
Alloc::Deallocate(p, n * sizeof(T));
}
} static void Deallocate(T *p)
{
Alloc::Deallocate(p, sizeof(T));
} static void SetCallBackHandle(CallBackFunc cb)
{
Alloc::SetCallBackHandle(cb);
}
};
simple_alloc.h
5.8. 测试
#include "stdio.h" #include "vector_define.h" #include<vector>
using namespace std; void main()
{
cout << "Vector(const size_type& nSize, const T& nValue)测试:" << endl;
Vector<int> vect1(, );
vect1.Print(); cout << "Vector()--Push()--Pop测试:" << endl;
Vector<int> vect2;
vect2.Push();
vect2.Push();
vect2.Push();
vect2.Push();
vect2.Print();
vect2.Pop_Back();
vect2.Print(); cout << "Iterator Erase(Iterator first, Iterator last)测试:" << endl;
Vector<int>::Iterator iter1 = vect2.Begin();
Vector<int>::Iterator iter2 = vect2.End();
vect2.Erase(iter1, iter2);
cout << "vect Size:" << vect2.Size() << endl; cout << "void Resize(size_type new_size, const T& x)测试:" << endl;
vect2.Resize(, );
vect2.Print();
vect2.Resize(, );
vect2.Print();
}
STL源码解析之vector自实现的更多相关文章
- Java集合类源码解析:Vector
[学习笔记]转载 Java集合类源码解析:Vector 引言 之前的文章我们学习了一个集合类 ArrayList,今天讲它的一个兄弟 Vector.为什么说是它兄弟呢?因为从容器的构造来说,Vec ...
- stl源码分析之vector
上篇简单介绍了gcc4.8提供的几种allocator的实现方法和作用,这是所有stl组件的基础,容器必须通过allocator申请分配内存和释放内存,至于底层是直接分配释放内存还是使用内存池等方法就 ...
- 《STL源码剖析》——Vector
vector vector的源码分为多个文件:vector.<<stl_vector.h>> vector的底层实现是在<<stl_vector.h>> ...
- STL源码剖析-vector
STL(Standard Template Library) C++标准模板库,acm选手一定对它不陌生,除了算法,那几乎是“吃饭的家伙了”.我们使用库函数非常方便,且非常高效(相对于自己实现来说). ...
- vector源码3(参考STL源码--侯捷):pop_back、erase、clear、insert
vector源码1(参考STL源码--侯捷) vector源码2(参考STL源码--侯捷):空间分配.push_back vector源码(参考STL源码--侯捷)-----空间分配导致迭代器失效 v ...
- vector源码(参考STL源码--侯捷):空间分配导致迭代器失效
vector源码1(参考STL源码--侯捷) vector源码2(参考STL源码--侯捷) vector源码(参考STL源码--侯捷)-----空间分配导致迭代器失效 vector源码3(参考STL源 ...
- vector源码2(参考STL源码--侯捷):空间分配、push_back
vector源码1(参考STL源码--侯捷) vector源码2(参考STL源码--侯捷) vector源码(参考STL源码--侯捷)-----空间分配导致迭代器失效 vector源码3(参考STL源 ...
- vector源码1(参考STL源码--侯捷):源码
vector源码1(参考STL源码--侯捷) vector源码2(参考STL源码--侯捷) vector源码(参考STL源码--侯捷)-----空间分配导致迭代器失效 vector源码3(参考STL源 ...
- Java 集合系列06之 Vector详细介绍(源码解析)和使用示例
概要 学完ArrayList和LinkedList之后,我们接着学习Vector.学习方式还是和之前一样,先对Vector有个整体认识,然后再学习它的源码:最后再通过实例来学会使用它.第1部分 Vec ...
随机推荐
- NoSQL2
系统的可扩展性是推动NoSQL运动发展的的主要理由,包含了分布式系统协调,故障转移,资源管理和许多其他特性.这么讲使得NoSQL听起来像是一个大筐,什么都能塞进去.尽管NoSQL运动并没有给分布式数据 ...
- HDU2385Stock
/*既然前一天的可以省下来给过后一天卖,那么就倒着卖,那么最后一天的只能在最后一天卖,如果可以卖完,还有可卖的名额,那么就从 倒数第二天取,如果卖不完只能丢掉,而且必定是尽量留到价格最高的那一天卖*/ ...
- java 标识符
java 所有的组成部分都需要名字.类名.变量名以及方法名都被称为标识符. java 标识符 1.所有的标识符都应该以字母(A-Z或者a-z),特殊符号(美元符$).或者下划线(_)开始 2.首字母之 ...
- Google揭露SHA-1碰撞,加速数据重删字节对比
原创 架构师技术联盟 近期,Google和道荷兰阿姆斯特研究者宣布攻破了世界上第一例公开的SHA-1哈希碰撞实例,业界一片哗然.当两组不同的数据(文件.一段数据)计算出相同的Hash值时,即视为二者 ...
- yum安装mysql后root用户的临时密码
1.查看root用户临时随机密码 yum 安装mysql后,无法通过空密码登录数据库,如下: [root@ mysql]# mysql -u root -p Enter password: ERROR ...
- Maven+Struts+Hibernate+Spring简单项目搭建
这段时间学习如何使用Maven+Struts+Hibernate+Spring注解方式建立项目,在这里感谢孙宇老师. 第一次写博客,主要是方便自己查看,同时分享给大家,希望对大家有所帮助,我也是 ...
- scala学习手记27 - 下划线与参数
在Scala里,下划线(_)可以表示函数值的参数.如果某个参数在函数里仅使用一次,就可以用下划线表示.每次在函数里用下划线,都表示随后的参数. val arr = Array(1, 2, 3, 4, ...
- HTML5 ——web audio API 音乐可视化(一)
使用Web Audio API可以对音频进行分析和操作,最终实现一个音频可视化程序. 最终效果请戳这里; 完整版代码请戳这里,如果还看得过眼,请给一个start⭐ 一.API AudioContext ...
- angular一些常用的方法:
angular.copy(); 用法:对Object对象的深度拷贝$scope.data = {name:'yanjinyun',age:'11'}; $scope.origData = angula ...
- 在UIElement外面多套一层布局面板(Grid、StackPanel)的意义
在一个UIElement或多个UIElement外面套上一层布局面板(Grid.StackPanel),可以起到统一管理作用(非重点关注):另外,更重要的是:可以起到扩大UIElement操作有效范围 ...