0

0

C++如何自定义内存分配器(allocator)_C++内存管理与自定义分配器实现

穿越時空

穿越時空

发布时间:2025-11-10 19:13:02

|

908人浏览过

|

来源于php中文网

原创

自定义内存分配器用于控制C++容器内存分配行为,通过实现标准接口如allocate、deallocate、construct等,可优化性能,常用于内存池、共享内存等场景,提升高频分配效率。

c++如何自定义内存分配器(allocator)_c++内存管理与自定义分配器实现

在C++中,自定义内存分配器(allocator)主要用于控制容器(如std::vectorstd::list等)的内存分配行为。标准库中的容器都支持通过模板参数传入自定义的allocator,从而实现更高效的内存管理,比如用于内存池、共享内存、上分配或性能优化场景。

理解标准分配器接口

C++标准要求一个合法的分配器必须满足一定的接口规范。最基本的成员包括:

  • value_type:被分配对象的类型
  • pointer:指向value_type的指针
  • const_pointer:常量指针
  • reference:引用类型
  • const_reference:常量引用
  • size_type:无符号整数类型,表示大小
  • difference_type:有符号整数类型,表示指针差值
  • allocate(n):分配未初始化的内存,可容纳n个value_type
  • deallocate(p, n):释放由allocate分配的内存
  • construct(p, args...):在已分配内存p上构造对象
  • destroy(p):析构p指向的对象
  • rebind:允许分配器适配不同类型

从C++17起,constructdestroy不再是强制要求,而是推荐使用std::allocator_traits来统一调用。

实现一个简单的自定义分配器

下面是一个基于堆的简单分配器示例,功能类似std::allocator,但便于扩展:

立即学习C++免费学习笔记(深入)”;

template 
struct MyAllocator {
    using value_type = T;
    using pointer = T*;
    using const_pointer = const T*;
    using reference = T&;
    using const_reference = const T&;
    using size_type = std::size_t;
    using difference_type = std::ptrdiff_t;
template zuojiankuohaophpcntypename Uyoujiankuohaophpcn
struct rebind {
    using other = MyAllocatorzuojiankuohaophpcnUyoujiankuohaophpcn;
};

MyAllocator() = default;

template zuojiankuohaophpcntypename Uyoujiankuohaophpcn
MyAllocator(const MyAllocatorzuojiankuohaophpcnUyoujiankuohaophpcn&) {}

pointer allocate(size_type n) {
    if (n == 0) return nullptr;
    pointer p = static_castzuojiankuohaophpcnpointeryoujiankuohaophpcn(::operator new(n * sizeof(T)));
    // 可添加日志或统计
    return p;
}

void deallocate(pointer p, size_type n) {
    ::operator delete(p);
    // 可记录释放信息
}

template zuojiankuohaophpcntypename U, typename... Argsyoujiankuohaophpcn
void construct(U* p, Args&&... args) {
    new(p) U(std::forwardzuojiankuohaophpcnArgsyoujiankuohaophpcn(args)...);
}

template zuojiankuohaophpcntypename Uyoujiankuohaophpcn
void destroy(U* p) {
    p-youjiankuohaophpcn~U();
}

};

这个分配器可用于STL容器:

PPT.AI
PPT.AI

AI PPT制作工具

下载
std::vector> vec;
vec.push_back(10);
vec.push_back(20);

高级用途:内存池分配器

实际应用中,自定义分配器常用于减少频繁调用new/delete带来的开销。例如,实现一个简单的内存池:

template 
class PoolAllocator {
private:
    struct Block {
        std::aligned_storage_t data;
    };
union Node {
    T data;
    Node* next;
};

Node* free_list = nullptr;
std::vectorzuojiankuohaophpcnBlockyoujiankuohaophpcn blocks;
size_t current_offset = 0;

public: using value_type = T; using pointer = T; using const_pointer = const T; using size_type = std::size_t;

template zuojiankuohaophpcntypename Uyoujiankuohaophpcn
struct rebind {
    using other = PoolAllocatorzuojiankuohaophpcnU, BlockSizeyoujiankuohaophpcn;
};

PoolAllocator() = default;

~PoolAllocator() = default;

pointer allocate(size_type n) {
    if (n != 1) {
        throw std::bad_alloc{};
    }
    if (!free_list) {
        refill_pool();
    }
    Node* p = free_list;
    free_list = free_list-youjiankuohaophpcnnext;
    return reinterpret_castzuojiankuohaophpcnpointeryoujiankuohaophpcn(p);
}

void deallocate(pointer p, size_type n) {
    if (n != 1 || !p) return;
    Node* node = reinterpret_castzuojiankuohaophpcnNode*youjiankuohaophpcn(p);
    node-youjiankuohaophpcnnext = free_list;
    free_list = node;
}

template zuojiankuohaophpcntypename U, typename... Argsyoujiankuohaophpcn
void construct(U* p, Args&&... args) {
    new(p) U(std::forwardzuojiankuohaophpcnArgsyoujiankuohaophpcn(args)...);
}

template zuojiankuohaophpcntypename Uyoujiankuohaophpcn
void destroy(U* p) {
    p-youjiankuohaophpcn~U();
}

private: void refill_pool() { Block& block = blocks.emplace_back(); char start = reinterpret_cast>(&block.data); const size_t nodes_per_block = BlockSize / sizeof(Node); Node nodes = reinterpret_cast>(start);

    for (size_t i = 0; i zuojiankuohaophpcn nodes_per_block - 1; ++i) {
        nodes[i].next = &nodes[i + 1];
    }
    nodes[nodes_per_block - 1].next = nullptr;

    free_list = &nodes[0];
}

};

该分配器预先分配大块内存,并将对象以链表形式管理空闲节点,显著提升小对象频繁分配/释放的性能。

注意事项与最佳实践

编写自定义分配器时需注意以下几点:

  • 确保allocate返回的是未初始化的原始内存,不要调用构造函数
  • deallocate不应调用析构,只负责释放内存
  • 分配器应为无状态(stateless)以便拷贝赋值安全;若需状态(如内存池),要保证线程安全
  • 不同实例间的分配器应能相互释放内存(即“可互换”),否则可能导致未定义行为
  • 避免在allocate/deallocate中抛出异常,除非确实无法分配

使用std::allocator_traits可以增强分配器的通用性,它提供了默认的constructdestroy实现,兼容旧式和新式分配器。

基本上就这些。自定义分配器是C++高性能编程的重要工具,合理使用可显著改善程序性能,尤其在高频分配场景下。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1434

2023.10.24

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

519

2023.09.20

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

173

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

92

2025.11.27

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

989

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

49

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

193

2025.12.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

366

2023.07.18

桌面文件位置介绍
桌面文件位置介绍

本专题整合了桌面文件相关教程,阅读专题下面的文章了解更多内容。

0

2025.12.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
HTML5/CSS3/JavaScript/ES6入门课程
HTML5/CSS3/JavaScript/ES6入门课程

共102课时 | 6.5万人学习

前端基础到实战(HTML5+CSS3+ES6+NPM)
前端基础到实战(HTML5+CSS3+ES6+NPM)

共162课时 | 18.5万人学习

第二十二期_前端开发
第二十二期_前端开发

共119课时 | 12.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号