0

0

c++怎么编写一个自定义的STL分配器_c++自定义allocator内存分配器实现

穿越時空

穿越時空

发布时间:2025-10-23 16:07:01

|

163人浏览过

|

来源于php中文网

原创

自定义STL分配器可控制C++容器内存行为,用于性能优化或内存池管理。需满足接口要求:定义value_type、allocate/deallocate分配内存、construct/destroy处理对象构造析构,支持rebind适配类型。示例中MyAllocator重载new/delete并打印日志,应用于vector时触发分配信息输出,实现简单但完整。高级场景可用内存池减少系统调用,提升频繁小对象分配效率,适用于游戏或高频交易系统。

c++怎么编写一个自定义的stl分配器_c++自定义allocator内存分配器实现

在C++中,自定义STL分配器(Allocator)可以让你控制容器的内存分配行为。比如用于性能优化、内存池管理或调试内存泄漏。标准库容器如 std::vectorstd::list 等都支持通过模板参数传入自定义分配器。

自定义分配器的基本要求

一个符合STL规范的分配器需要满足一些基本接口要求。虽然C++17后对分配器的要求有所简化,但核心成员仍然包括:

  • value_type:被分配对象的类型
  • pointer:指向value_type的指针
  • const_pointer:常量指针
  • reference:引用类型
  • const_reference:常量引用
  • size_type:大小类型(通常是 size_t)
  • difference_type:指针差值类型
  • allocate(n):分配n个对象的空间(不构造)
  • deallocate(p, n):释放从p开始的n个对象空间(不析构)
  • construct(ptr, args...):在指定位置构造对象(C++17起可选)
  • destroy(ptr):显式调用析构函数(C++17起可选)
  • rebind:允许分配器适配不同类型(旧标准需要,C++11以后可用别名模板替代)

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

下面是一个通用的自定义分配器示例,使用全局 ::operator new::operator delete,但你可以替换成内存池或其他机制。

零一万物开放平台
零一万物开放平台

零一万物大模型开放平台

下载

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

#include 
#include 
#include 

template class MyAllocator { public: 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;

// C++17 起使用 type alias 替代 rebind
templatezuojiankuohaophpcntypename Uyoujiankuohaophpcn
struct rebind {
    using other = MyAllocatorzuojiankuohaophpcnUyoujiankuohaophpcn;
};

// 构造函数(必须提供默认构造)
MyAllocator() noexcept = default;

// 支持不同类型的转换构造(STL可能用到)
templatezuojiankuohaophpcntypename Uyoujiankuohaophpcn
MyAllocator(const MyAllocatorzuojiankuohaophpcnUyoujiankuohaophpcn&) noexcept {}

// 分配原始内存,不构造对象
pointer allocate(size_type n) {
    std::cout zuojiankuohaophpcnzuojiankuohaophpcn "Allocating " zuojiankuohaophpcnzuojiankuohaophpcn n zuojiankuohaophpcnzuojiankuohaophpcn " elements of size " zuojiankuohaophpcnzuojiankuohaophpcn sizeof(T) zuojiankuohaophpcnzuojiankuohaophpcn std::endl;
    if (n == 0) return nullptr;
    pointer p = static_castzuojiankuohaophpcnpointeryoujiankuohaophpcn(::operator new(n * sizeof(T)));
    return p;
}

// 释放内存,不调用析构
void deallocate(pointer p, size_type n) noexcept {
    std::cout zuojiankuohaophpcnzuojiankuohaophpcn "Deallocating " zuojiankuohaophpcnzuojiankuohaophpcn n zuojiankuohaophpcnzuojiankuohaophpcn " elements" zuojiankuohaophpcnzuojiankuohaophpcn std::endl;
    ::operator delete(p);
}

// 构造对象(C++17 推荐实现)
templatezuojiankuohaophpcntypename U, typename... Argsyoujiankuohaophpcn
void construct(U* p, Args&&... args) {
    new(p) U(std::forwardzuojiankuohaophpcnArgsyoujiankuohaophpcn(args)...);
}

// 析构对象
templatezuojiankuohaophpcntypename Uyoujiankuohaophpcn
void destroy(U* p) {
    p-youjiankuohaophpcn~U();
}

// 比较两个分配器是否相等(一般无状态分配器返回true)
bool operator==(const MyAllocator&) const { return true; }
bool operator!=(const MyAllocator&) const { return false; }

};

// 非成员函数(可选) template bool operator==(const MyAllocator& a, const MyAllocator& b) { return true; }

template bool operator!=(const MyAllocator& a, const MyAllocator& b) { return false; }

使用自定义分配器

将上面的分配器用于 std::vector

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

int main() {
    std::vector> vec;
vec.push_back(10);
vec.push_back(20);
vec.push_back(30);

for (const auto& v : vec) {
    std::cout zuojiankuohaophpcnzuojiankuohaophpcn v zuojiankuohaophpcnzuojiankuohaophpcn " ";
}
std::cout zuojiankuohaophpcnzuojiankuohaophpcn std::endl;

return 0;

}

输出示例:

Allocating 1 elements of size 4
Allocating 2 elements of size 4
Allocating 4 elements of size 4
10 20 30 
Deallocating 4 elements

高级用途:内存池分配器

如果你希望进一步提升性能,可以实现基于内存池的分配器。例如预先分配一大块内存,allocate 时从中切分,避免频繁系统调用。

关键点:

  • 维护一个空闲链表
  • 重载 allocate/deallocate 使用内部缓冲区
  • 注意线程安全(如需多线程使用)
  • 确保对齐(使用 alignasstd::aligned_storage

这种分配器适合频繁小对象分配的场景,比如游戏引擎或高频交易系统。

基本上就这些。自定义分配器不复杂,但容易忽略细节,尤其是构造/析构语义和比较操作。只要遵循STL规范,就能安全集成到标准容器中。

相关专题

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

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

1492

2023.10.24

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

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

527

2023.09.20

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

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

527

2023.09.20

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

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

1051

2023.10.19

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

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

107

2025.10.17

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

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

697

2025.12.29

java接口相关教程
java接口相关教程

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

11

2026.01.19

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

482

2023.08.10

c++空格相关教程合集
c++空格相关教程合集

本专题整合了c++空格相关教程,阅读专题下面的文章了解更多详细内容。

0

2026.01.23

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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