0

0

C++如何实现带缓存的函数结果?(memoization模式)

尼克

尼克

发布时间:2026-02-22 17:19:03

|

993人浏览过

|

来源于php中文网

原创

手动实现 memoization 需用 std::map 或 std::unordered_map 管理缓存,注意键类型需支持比较或哈希、缓存生命周期须覆盖所有调用、递归中检查必须在调用前、线程安全需加锁或读写锁。

c++如何实现带缓存的函数结果?(memoization模式)

std::mapstd::unordered_map 手动缓存结果最直接

手动实现 memoization 的核心是:函数第一次调用时计算并存入缓存,后续相同参数直接返回缓存值。C++ 没有内置装饰器,所以得自己管键构造、查表、回填。

常见错误是把参数直接当 key 用——比如传入 std::vector 或自定义类型却没提供 operator(对 <code>std::map)或哈希函数(对 std::unordered_map),导致编译失败或行为未定义。

  • 简单参数组合(如两个 int)推荐拼成 std::pair<int int></int>,它自带比较和哈希(C++17 起)
  • 字符串参数优先用 std::string 作 key,别用 C 风格指针
  • 避免在递归函数里把缓存对象声明为局部静态——它会随每次调用重建,起不到跨调用缓存作用

示例:斐波那契带缓存

int fib(int n, std::unordered_map<int, int>& cache) {
    if (n <= 1) return n;
    if (cache.find(n) != cache.end()) return cache[n];
    cache[n] = fib(n-1, cache) + fib(n-2, cache);
    return cache[n;
}

std::function + lambda 实现通用缓存包装器

想复用缓存逻辑?可以写一个模板函数,接收原函数对象,返回带缓存的新函数。但要注意:lambda 捕获的缓存容器生命周期必须覆盖所有调用,否则缓存失效甚至崩溃。

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

典型坑是返回局部 lambda,里面捕获了局部 std::unordered_map —— 函数返回后 map 被析构,后续调用访问野指针。

畅图
畅图

AI可视化工具

下载
  • 安全做法是让缓存作为返回 lambda 的 mutable 捕获变量,或用 std::shared_ptr 管理
  • 不能用于带重载或模板函数名(如 std::abs),必须先绑定成具体 std::function
  • 参数包展开要小心引用折叠,const T&& 可能意外延长临时对象生命,也可能截断 const 性质

简版示意:

template<typename F>
auto memoize(F f) {
    std::unordered_map<std::tuple<int>, int> cache; // 单参 int 版本
    return [f, cache = std::move(cache)](int x) mutable -> int {
        auto key = std::make_tuple(x);
        auto it = cache.find(key);
        if (it != cache.end()) return it->second;
        int res = f(x);
        cache[key] = res;
        return res;
    };
}

递归函数里缓存位置不对会导致无限递归或漏缓存

在递归函数内部做缓存检查,必须放在递归调用之前;否则每次递归都重新算,缓存只对最外层有效。

另一个易错点:缓存键没包含全部影响结果的参数。比如函数实际依赖全局状态或某个隐藏上下文,但 key 只用了显式参数,结果缓存命中却返回错误值。

  • 检查是否所有输入变量都参与了 key 构造(包括隐式 this 指针,如果方法是非 static)
  • 避免用指针地址作 key——同一数据不同地址算不同 key,失去缓存意义
  • 若函数有副作用(如打印日志),缓存后需确认是否仍需执行副作用,否则行为不一致

注意线程安全和移动语义的冲突

多个线程同时调用同一个缓存函数,std::unordered_map::operator[]insert 都不是线程安全的。加锁会拖慢性能,但不加锁可能崩溃或读到脏数据。

更隐蔽的问题是:缓存值类型支持移动但不支持拷贝(比如 std::unique_ptr),而 std::map::operator[] 在 key 不存在时会默认构造再赋值,触发移动构造没问题;但某些封装逻辑若误用 cache.at(key)(抛异常)或手动 find 后解引用,就可能绕过默认构造路径,导致未定义行为。

  • 多线程场景下,优先用 std::shared_mutex(C++17)读写锁,而非粗粒度 std::mutex
  • 缓存值类型尽量保持 trivially copyable,避免移动语义引入生命周期歧义
  • 不要假设 cache[key] 总是“存在即返回”,它在不存在时会插入默认值——这本身也是副作用
缓存键的设计和生命周期管理,比写几行 if (cache.count(k)) 容易得多;多数 bug 都出在这两处,而不是算法逻辑本身。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

810

2023.08.02

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

826

2023.08.22

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

199

2023.11.20

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

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

550

2023.09.20

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

616

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

217

2023.09.04

java基础知识汇总
java基础知识汇总

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

1557

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

642

2023.11.24

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

1030

2026.02.13

热门下载

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

精品课程

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

共94课时 | 10万人学习

C 教程
C 教程

共75课时 | 4.9万人学习

C++教程
C++教程

共115课时 | 18.9万人学习

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

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