0

0

c++的std::pmr (多态内存资源) 如何自定义内存分配器? (C++17)

尼克

尼克

发布时间:2026-01-15 15:27:09

|

713人浏览过

|

来源于php中文网

原创

c++kquote>std::pmr::memory_resource 是一个抽象基类,定义了 do_allocate、do_deallocate、do_is_equal 三个纯虚函数,不可实例化,所有 std::pmr 容器通过其指针和虚函数调用内存操作;自定义实现需继承并正确实现三函数,尤其 do_is_equal 必须严格比较对象地址。

c++的std::pmr (多态内存资源) 如何自定义内存分配器? (c++17)

std::pmr::memory_resource 是什么,为什么不能直接 new/delete

它是一个抽象基类,定义了 do_allocatedo_deallocatedo_is_equal 三个纯虚函数。你不能实例化它,也不能靠重载 operator new 来“替换”它——std::pmr 的所有容器(如 std::pmr::vector)只认 std::pmr::memory_resource* 指针,且全程通过虚函数调用,不依赖全局或类作用域new

写一个最简可用的自定义 memory_resource

核心是继承 std::pmr::memory_resource,实现三个虚函数。注意:do_is_equal 必须判断是否为同一对象(不是“等价”,而是地址相等),否则 std::pmr::polymorphic_allocator 可能跳过缓存或误判资源归属。

class SimplePoolResource : public std::pmr::memory_resource {
private:
    std::vector pool_;
    size_t offset_ = 0;

protected: void do_allocate(size_t bytes, sizet alignment) override { if (bytes == 0) return nullptr; auto ptr = std::align(alignment, bytes, pool.data() + offset, pool.size() - offset_); if (!ptr) throw std::badalloc{}; offset = static_cast>(ptr) - pool_.data() + bytes; return ptr; }

void do_deallocate(void* p, size_t bytes, size_t alignment) override {
    // 简单池不回收,什么也不做
}

bool do_is_equal(const std::pmr::memory_resource& other) const noexcept override {
    return this == &other; // 必须是同一对象
}

public: explicit SimplePoolResource(sizet capacity) : pool(capacity) {} };

这个实现只支持一次性分配(类似 arena),不支持释放单块内存,但完全合法——std::pmr 不要求 deallocate 真正归还空间。

如何让 std::pmr::vector 用上你的 resource

关键不是传 resource 给 vector 构造函数,而是通过 std::pmr::polymorphic_allocator 包装后传入。vector 自身不持有 resource 指针,allocator 才持有。

Shakespeare
Shakespeare

一款人工智能文案软件,能够创建几乎任何类型的文案。

下载

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

  • std::pmr::vector 是类型别名,等价于 std::vector>
  • 构造时必须显式传入 allocator,否则默认使用 std::pmr::get_default_resource()
  • resource 生命周期必须长于所有使用它的容器
SimplePoolResource pool{1024 * 1024};
std::pmr::polymorphic_allocator alloc{&pool};
std::pmr::vector v{alloc}; // 正确:v 的所有内存来自 pool
v.reserve(1000); // 分配成功
// pool 析构前,v 不能继续 allocate

常见陷阱和兼容性注意点

C++17 标准库std::pmr 实现程度不一;libstdc++(GCC)直到 GCC 10 才完整支持,MSVC 2019 Update 3 起稳定,Clang 需搭配 libc++ 7.0+。更隐蔽的问题是:

  • 某些 STL 容器内部会额外调用 resource()->is_equal(other->resource()) 做优化,若 do_is_equal 返回错误结果,可能导致未定义行为
  • std::pmr::synchronized_pool_resourcestd::pmr::unsynchronized_pool_resource 是标准提供的可复用实现,但它们内部仍依赖 upstream_resource(),若你传入的 upstream 是临时对象,极易悬垂
  • 调试时无法直接 print resource 地址来比对,因为 std::pmr::polymorphic_allocatorresource() 返回的是 const 指针,且 operator== 默认比较的是 resource 对象本身而非指针值

真正难的不是写 allocate/deallocate,而是保证 resource 对象的生存期、线程安全(如果多线程用)和 is_equal 的语义正确性。这三个点出错,往往表现为偶发崩溃或静默内存泄漏,而不是编译报错。

相关专题

更多
python中print函数的用法
python中print函数的用法

python中print函数的语法是“print(value1, value2, ..., sep=' ', end=' ', file=sys.stdout, flush=False)”。本专题为大家提供print相关的文章、下载、课程内容,供大家免费下载体验。

184

2023.09.27

resource是什么文件
resource是什么文件

Resource文件是一种特殊类型的文件,它通常用于存储应用程序或操作系统中的各种资源信息。它们在应用程序开发中起着关键作用,并在跨平台开发和国际化方面提供支持。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

149

2023.12.20

java多态详细介绍
java多态详细介绍

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

15

2025.11.27

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

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

523

2023.09.20

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

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

523

2023.09.20

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

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

480

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

143

2025.12.24

数据库Delete用法
数据库Delete用法

数据库Delete用法:1、删除单条记录;2、删除多条记录;3、删除所有记录;4、删除特定条件的记录。更多关于数据库Delete的内容,大家可以访问下面的文章。

269

2023.11.13

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

8

2026.01.15

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 6.8万人学习

C 教程
C 教程

共75课时 | 4万人学习

C++教程
C++教程

共115课时 | 12.3万人学习

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

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