STL——空间配置器(SGI-STL)

2023-12-12 04:32
文章标签 配置 空间 stl sgi

本文主要是介绍STL——空间配置器(SGI-STL),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、 空间配置器标准接口

参见《STL源码剖析》第二章-2.1。<memory>文件。

二、具备次配置力的SGI空间配置器

1. SGI STL的配置器与众不同,也与标准规范不同,其名称是alloc而非allocator,而且不接受任何参数(虽然SGI也定义有一个符合部分标准、名为sllocator的配置器,但SGI自己从未用过它,也不建议使用,主要因为效率不佳,它只是基层内存配置/释放行为(也就是::operator new和 ::operator delete)的一层薄薄的包装,并没有考虑到任何效率上的强化)。这并不会带来什么困扰:我们通常很少需要自行指定配置器名称,而SGI STL的每一个容器都已经指定其缺省的空间配置器为alloc。

复制代码
// 在程序中要明白采用SGI配置器,则不能采用标准写法:
vector <int, std::allocator<int> > iv;    // 标准写法,in VC or CB
vector <int, std::alloc> iv;        // SGI,in GCC// SGI STL 每一个容器都已经指定缺省空间配置器
template <class T, class Alloc = alloc >     // 缺省使用alloc为配置器
class vector { ... };
复制代码

 2. SGI特殊的空间配置器——std::alloc

一般而言,我们所习惯的C++内存配置操作和释放操作是这样的:

class Foo { ... };
Foo* pf = new Foo;       // 配置内存,然后构造对象
delete pf;            // 将对象析构,然后释放内存

这其中的new算式内含两阶段操作:(1)调用::operator new 配置内存,(2)调用Foo::Foo() 构造对象内容。delete算式也内含两阶段操作:(1)调用Foo::~Foo() 将对象析构;(2)调用 ::operator delete 释放内存。
为了精密分工,STL allocator 决定将这两个阶段操作区分开来。内存配置操作由alloc::allocate()负责,内存释放操作由 alloc::deallocate() 负责;对象构造操作由 ::construct()负责,对象析构操作由::destroy()负责。STL配置器定义于<memory>之中,实现在于内含的<stl_alloc.h> 和 <stl_construct.h> 两个文件之中。

3. 构造和析构基本工具:construct() 和 destroy()

上述construct()接受一个指针p和一个初值value,该函数的用途就是将初值设定到指针所指的空间上。C++的placement new 运算子可用来完成这一任务。

destroy()有两个版本,第一版本接受一个指针,准备将该指针所指之物析构掉。这很简单,直接调用该对象的析构函数即可。第二版本接受first和last两个迭代器,准备将[ first, last )范围内的所有对象析构掉(注意,这是一个左闭右开的范围)。如果范围很大,而每个对象的析构函数都无关痛痒(所谓trivial destructor),那么一次次调用这些无关痛痒的析构函数,对效率是一种伤害。因此,这里首先利用value_type()获得迭代器所指对象的型别,再利用__type_traits<T>判断该型别的析构函数是否无关痛痒。若是(__true_type),则什么也不做就结束;若否(__false_type),这才以循环方式巡防整个范围,并在循环中每经历一个对象就调用第一个版本的destroy()。 (上述value_type()和__type_traits<>在《STL源码剖析》3.7节有详细介绍。)

4. 空间的配置和释放,std::alloc

对象构造前的空间配置和对象析构后的空间释放,由<stl_alloc.h>负责,SGI对此的设计哲学如下:  

  (1)向system heap 要求空间;  

  (2)考虑多线程(multi-threads)状态;  

  (3)考虑内存不足时的应变措施;  

  (4)考虑过多“小型区块”可能造成的内存碎片(fragment)问题。

C++的内存配置基本操作是::operator new(),内存释放基本操作是::operator delete()。这两个全局函数相当于C的malloc() 和 free() 函数。SGI正是以malloc()和free() 完成内存的配置和释放。考虑到小型区块所可能造成的内存破碎问题,SGI 设计了双层级配置器,第一级配置器直接使用malloc() 和 free() ,第二级配置器则是情况采用不同的策略:以配置128bytes区块为界,大于则调用第一级配置器,小于则采用复杂的memory pool整理方式,同时也取决是否定义了_USE_MALLOC。

复制代码
#ifdef _USE_MALLOC
...
typedef __malloc_alloc_template<0> malloc_alloc;
typedef malloc_alloc alloc;           // 令alloc为第一级配置器
#else
...
// 令alloc为第二级配置器
typedef __default_alloc_template<__NODE_ALLOCATOR_THREADS, 0> alloc;
#endif     /* ! _USE_MALLOC*/
复制代码

其中__malloc_alloc_template就是第一级配置器,__default_alloc_template就是第二级配置器。注意:alloc并不接受任何template参数。

无论alloc被定义为第一级配置器或第二级配置器(SGI STL容器缺省使用第二级配置器),SGI还为它再包装一个接口如下,使配置器的接口能够符合STL规格:

复制代码
// 其内部的四个成员函数其实都是单纯的转调用,调用传递给配置器的成员函数
template<class T, class Alloc>
class simple_alloc {
public:static T *allocate(size_t n){ return 0 == n ? 0 : (T*)Alloc::allocate(n * sizeof(T)); }static T *allocate(void){ return (T*)Alloc::allocate(n * sizeof(T)); }static T *deallocate(T *p, size_t n){ if( 0 != n) ? 0 : (T*)Alloc::deallocate(p, n * sizeof(T)); }static T *allocate(size_t n){ Alloc::deallocate(p, sizeof(T)); }};
复制代码

SGI STL容器全都使用这个simple_alloc接口:

复制代码
template <class T, class Alloc = alloc>      // 缺省使用alloc为配置器
class vector{
protected:// 专属之空间配置器,每次配置一个元素大小typedef simple_alloc<value_type, Alloc> data_allocator;void deallocte() {if ( ... )data_allocator::deallocate(start, end_of_storage - start);}...
};
复制代码

5. 第一级配置器 __malloc_alloc_template 剖析 第一级配置器以malloc(), free(), realloc() 等C函数执行实际的内存配置、释放、重配置操作,并实现出类型C++ new-handler的机制。是的,它不能直接运用C++ new-handler机制,因为它并非使用::operator new来配置内存。注意,它没有“template型别参数”。参见相关源码。

6. 第二级配置器 __default_alloc_template剖析 第二级配置器多了一些机制,避免太多小额区块造成内存的碎片和配置时的额外负担。SGI第二级配置器的做法是:如果区块够大,超过128bytes时,就移交第一级配置器处理。当区块小于128bytes时,则以内存池(memory pool)管理,此法又称为次层配置(sub-allocation):每次配置一大块内存,并维护对应之自由链表(free-list)。下次若再有相同大小的内存需求,就直接从free-list中拨出。如果客户端释还小额区块,就由配置器回收到free-list中——是的,别忘了,配置器除了负责配置,也负责回收。为了管理方便,SGI第二级配置器会主动将任何小额区块的内存需求量上调至8的倍数(例如客户端要求30bytes,就自动调整为32bytes),并维护16个free-list,各自管理大小分别为8, 16, 24, 32, 40, 48, 56, 64, 72, 80, 88, 96, 104, 112, 120, 128bytes的小额区块。free-list的节点结构如下:

复制代码
// 使用union类型,不会为了维护链表所必须的指针而造成内存的另一种浪费
union obj
{union obj * free_list_link;char client_data[1];        // the client sees this
};
复制代码

参见相关源码。

7. 空间配置函数allocate()

8. 空间释放函数 deallocate()

9. 重新填充free lists 当发现free-list中没有可用区块了时,就调用refill(),准备为free list重新填充空间。新的空间将取自内存池(经由chunk_alloc完成)。缺省取得20个新节点(新区块),但万一内存池空间不足,获得的节点数(区块数)可能小于20。参见相关源码。

10. 内存池(memory pool) 从内存池中取空间给free list 使用,是chunk_alloc() 的工作。参见相关源码。 chunk_alloc()函数以end_free - start_free来判断内存池的水量。如果水量充足,就直接调出20个区块返回给free list。如果水量不足以提供20个区块,但还足够供应一个以上的区块,就拨出这不足20个区块的空间出去。这时候其pass by reference 的nobjs 参数将被修改为实际能够供应的区块数。如果内存池连一个区块空间都无法供应,对客户端显然无法交待,此时便需利用malloc()从heap中配置内存,为内存池注入源头活水以应付需求。新水量的大小为需求量的两倍,再加上一个随着配置次数增加而愈来愈大的附加量。

万一,整个system heap空间都不够了(以至于无法为内存池注入源头活水),malloc() 行动失败,chunk_alloc() 就四处寻找有无“尚有未用区块,且区块够大”之free list。找到了就挖一块交出,找不到就调用第一级配置器。第一级配置器其实也是使用malloc()来配置内存,但它有out-of-memory处理机制(类似new-handler机制),或许有机会释放其他的内存拿来此处用。如果可以,就成功,否则发出bad-alloc异常。

三、内存基本处理工具
STL 定义有五个全局函数,作用于未初始化空间上。这样的功能对于容器的实现很有帮助,在《STL源码剖析》第4章容器实现代码中,看到它们肩负的重任。前两个函数是前面说过的、用于构造的construct() 和用于析构的 destory() ,另三个函数是 uninitialized_copy(), uninitialized_fill(), uninitializd_fill_n(),分别对应于高层次函数copy(), fill(), fill_n()——这些都是STL算法,在第6章介绍。如果要使用本节的三个低层次函数,应该包含<memory>,不过SGI 把它们实际定义于<stl_uninitialized>。

1. uninitialized_copy

template <class InputIterator, class ForwardIterator>
ForwardIterator
uninitialized_copy( InpuIterator first, InpuIterator last, ForwardIterator result);

uninitialized_copy() 使我们能够将内存的配置与对象的构造行为分离开来。如果作为输出目的地的[ result, result + (last - first))范围内的每一个迭代器都指向未初始化区域,则uninitialized_copy() 会使用copy constructor,给身为输入来源之[first, last) 范围内的每一个对象产生一份复制品,放进输出范围中。(使用上面的construct 构造工具)。也就是说,针对输入范围内的每一个迭代器i,该函数会调用 construct( &*(result+(i-first)), *i ), 产生 *i 的复制品,放置于输出范围的相对位置上。

这是一个非常有用的工具,因为容器的全区间构造函数通常以两个步骤完成:

 (1)配置内存区块,足以包含范围内的所有元素。
 (2)使用uninitialized_copy() ,在该内存区块上构造元素。

2. uninitialized_fill

template <class FrowardIterator, class T>
void uninitialized_fill(FrowardIterator first, FrowardIterator last, const T& x);

uninitialized_fill() 也能够使我们将内存配置与对象的构造行为分离开来。如果[ first, last ) 范围内的每个迭代器都指向未初始化的内存,那么uninitialized_fill() 会在该范围内产生x(上式第三参数)的复制品。

注意:与uninitialized_copy() 一样,uninitialized_fill() 必须具备 “commit or rollback”语意,换句话说,它要么产生出所有必要元素,要么不产生任何元素(异常安全等级)。如果有任何一个copy constructor 丢出异常,uninitialized_fill 必须能够将已产生的所有元素析构掉。

3. uninitialized_fill_n

template <class ForwardIterator, class Size, class T>
ForwardIterator
uninitialized_fill_n(ForwardIterator first, Size n, const T& x);

uninitialized_fill_n() 能够使我们将内存配置与对象构造行为分离开来。它会为指定范围内的所有元素设定相同的初值。

如果[ first, first+n )范围内的每一个迭代器都指向未初始化的内存,那么 uninitialized_fill_n() 会调用copy constructor ,在该范围内产生x(上式第三参数)的复制品。uninitialized_fill_n() 也具有 “commit or rollback”语意。

这三个函数的实现法参见相关源码。其中所呈现的 iterators(迭代器)、value_type()、 __type_traits、__true_type、__false_type、is_POD_type等实现技术,在《STL源码剖析》都有详细介绍。

这篇关于STL——空间配置器(SGI-STL)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/483237

相关文章

windos server2022的配置故障转移服务的图文教程

《windosserver2022的配置故障转移服务的图文教程》本文主要介绍了windosserver2022的配置故障转移服务的图文教程,以确保服务和应用程序的连续性和可用性,文中通过图文介绍的非... 目录准备环境:步骤故障转移群集是 Windows Server 2022 中提供的一种功能,用于在多个

windos server2022里的DFS配置的实现

《windosserver2022里的DFS配置的实现》DFS是WindowsServer操作系统提供的一种功能,用于在多台服务器上集中管理共享文件夹和文件的分布式存储解决方案,本文就来介绍一下wi... 目录什么是DFS?优势:应用场景:DFS配置步骤什么是DFS?DFS指的是分布式文件系统(Distr

关于Maven中pom.xml文件配置详解

《关于Maven中pom.xml文件配置详解》pom.xml是Maven项目的核心配置文件,它描述了项目的结构、依赖关系、构建配置等信息,通过合理配置pom.xml,可以提高项目的可维护性和构建效率... 目录1. POM文件的基本结构1.1 项目基本信息2. 项目属性2.1 引用属性3. 项目依赖4. 构

龙蜥操作系统Anolis OS-23.x安装配置图解教程(保姆级)

《龙蜥操作系统AnolisOS-23.x安装配置图解教程(保姆级)》:本文主要介绍了安装和配置AnolisOS23.2系统,包括分区、软件选择、设置root密码、网络配置、主机名设置和禁用SELinux的步骤,详细内容请阅读本文,希望能对你有所帮助... ‌AnolisOS‌是由阿里云推出的开源操作系统,旨

mysql-8.0.30压缩包版安装和配置MySQL环境过程

《mysql-8.0.30压缩包版安装和配置MySQL环境过程》该文章介绍了如何在Windows系统中下载、安装和配置MySQL数据库,包括下载地址、解压文件、创建和配置my.ini文件、设置环境变量... 目录压缩包安装配置下载配置环境变量下载和初始化总结压缩包安装配置下载下载地址:https://d

gradle安装和环境配置全过程

《gradle安装和环境配置全过程》本文介绍了如何安装和配置Gradle环境,包括下载Gradle、配置环境变量、测试Gradle以及在IntelliJIDEA中配置Gradle... 目录gradle安装和环境配置1 下载GRADLE2 环境变量配置3 测试gradle4 设置gradle初始化文件5 i

SpringCloud配置动态更新原理解析

《SpringCloud配置动态更新原理解析》在微服务架构的浩瀚星海中,服务配置的动态更新如同魔法一般,能够让应用在不重启的情况下,实时响应配置的变更,SpringCloud作为微服务架构中的佼佼者,... 目录一、SpringBoot、Cloud配置的读取二、SpringCloud配置动态刷新三、更新@R

MySQL中my.ini文件的基础配置和优化配置方式

《MySQL中my.ini文件的基础配置和优化配置方式》文章讨论了数据库异步同步的优化思路,包括三个主要方面:幂等性、时序和延迟,作者还分享了MySQL配置文件的优化经验,并鼓励读者提供支持... 目录mysql my.ini文件的配置和优化配置优化思路MySQL配置文件优化总结MySQL my.ini文件

C#读取本地网络配置信息全攻略分享

《C#读取本地网络配置信息全攻略分享》在当今数字化时代,网络已深度融入我们生活与工作的方方面面,对于软件开发而言,掌握本地计算机的网络配置信息显得尤为关键,而在C#编程的世界里,我们又该如何巧妙地读取... 目录一、引言二、C# 读取本地网络配置信息的基础准备2.1 引入关键命名空间2.2 理解核心类与方法

最新版IDEA配置 Tomcat的详细过程

《最新版IDEA配置Tomcat的详细过程》本文介绍如何在IDEA中配置Tomcat服务器,并创建Web项目,首先检查Tomcat是否安装完成,然后在IDEA中创建Web项目并添加Web结构,接着,... 目录配置tomcat第一步,先给项目添加Web结构查看端口号配置tomcat    先检查自己的to