内存管理器及其策略

本文介绍了C++中两种主流的内存管理策略:通用意图分配策略和固定大小分配策略,并探讨了垃圾收集技术的应用。此外,文章还分析了不同层面的内存管理服务,包括操作系统、编译器运行时库、标准库容器及自定义分配器的角色。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 
概观
简而言之,C++中存在着如下两个流行的内存管理策略:
其一是通用意图(general-purpose)的分配策略,它可以提供调用方的任意大小(称为请求大小或块大小)的内存块,通用意图的分配策略非常灵活,但也有几个缺点,其中两个就是:(a)性能,因为这种分配策略需要做更多的工作;(b)碎片,因为随着内存块的分配和释放,最终可能导致许多小的,不连续的未分配的内存区域。
 
其二是固定大小(fixed-size)的分配策略,它总是返回固定大小的内存块,着显然不象通用意图的分配策略那样灵活,但它的好处是可以在得多的时间内完成分配,而且不会产生那样的内存碎片。
 
第三种主要就是垃圾收集,该策略与C/C++指针,mallocnew并不完全兼容,因此与我们当前讨论的主题不直接相关;然而,垃圾收集技术正在变得越来越流行,而且正在走进C++,只不过不是跟指针和new搭配。
 
时间上,我们通常会见到这些策略是被组合在一起运用的。例如,或许某些内存管理器对任何大于S的内存请求使用的都是通用意图的分配,而对小于等于S的请求册一律采用固定大小的分配策略以获得优化的效果。一般来说,为大小是1,是2,是3等等的内存请求全都特殊对待、的做法是不实用的,所以内存管理器一般会机遇某个特定大小(如16字节)的倍数维护一个个的所谓仓库(arena,譬如,现请求16字节的内存,那么很好,因为恰好只使用16字节;但若请求17字节的内存,这块内存就会从“存放”着32位字节大小的内存块的arena中分配出来,即浪费了15字节。这是内存管理策略可能导致的额外开销的来源之一。
 
划分策略
 C++标准库异己使用了C++标准库实现的典型环境中存放在有好几种层次的内存管理,其中每层都可能回覆盖原先的(较低的)层面:
 
首先是操作系统内核提供了最为基础的内存反陪服务。这个底层的分配策略及其特性可能根据操作系统平台的不同而有很大差异,而且这一层内存管理是最可能受到硬件相关的考虑所影响的。
 
 编译器的默认运行时库也回建立起它自己的内存分配服务,譬如C++中的opterator newC中的malloc,这一层是建立在操作系统的本地分配服务基础上的,可能只是后者的一层薄薄的外衣,继承了其特性;也有可能通过先从后者那里“买进”大块内存(chunk),然后再切割转手分配出去的方式来覆盖操作系统的本地内存分配服务,至于如何切割“买来的”大块内存就是由这一层运行时库自己自由选择的了。
 
 而标准库容器和标准分配器进而利用(也许覆盖)运行时库提供的服务来实现它们自己的策略和优化。
 
 最后,用户自己定义的同期或用户自定义的分配器可能会进一步复用上面提到的任何一层服务(例如,若可移植性并不呢们重要,它们可能会想要直接访问本地分配服务),并按照它们自己的策略和优化
 
因此,内存分配器有好些不同的形式和种类,而不同的操作系统上描绘相同的操作系统上的不同编译器,或不同容器之间都会有所不同,甚至即便是在不同的对象之间也会有所不同
 
准则:弄清楚谁负责做什么;弄清楚自己当前的平台和标准库所使用的实际(且系统相关的)分配策略以及他们各自的职责是什么。
 
现代操作系统撒谎功能的内存管理就其本身来说可能是复杂而成熟的,然而充其量它也不过是(仅仅)一层(值得C++程序员关心的)内存管理层而已。标准库则提供了更多层次的让我们选择;首先是它的用于分配和归还的基本设施,然后是它的标准容器和分配器,再然后就是自己编写的名目繁多的自定义容器了。
 
 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值