JavaScript 中如何实现函数缓存

news2024/10/23 8:17:40

在JavaScript中,函数缓存是一种优化技术,它通过将函数的计算结果存储起来,以避免在后续调用中重复计算相同的值。这特别适用于那些计算成本高昂且输入参数有限的函数。以下是实现函数缓存的几种常见方法:

1. 使用对象作为缓存

这是最直接的方法,通过创建一个对象来存储函数的输入和对应的输出。每次调用函数时,首先检查缓存中是否已有该输入的结果,如果有则直接返回,否则进行计算并存储结果。

function memoizedFunction(key) {
  const cache = memoizedFunction.cache = memoizedFunction.cache || {};

  if (cache[key]) {
    return cache[key];
  }

  // 假设这是你的计算逻辑
  const result = someExpensiveComputation(key);

  // 存储结果到缓存中
  cache[key] = result;

  return result;
}

在这个例子中,memoizedFunction具有一个名为cache的属性,用于存储输入和输出的映射。如果缓存中已经有对应输入的结果,则直接返回该结果。

2. 使用闭包和Map对象

使用闭包可以封装缓存逻辑,并且Map对象提供了一种更现代和灵活的方式来存储键值对。

function createMemoizedFunction() {
  const cache = new Map();

  return function(key) {
    if (cache.has(key)) {
      return cache.get(key);
    }

    const result = someExpensiveComputation(key);
    cache.set(key, result);

    return result;
  };
}

const memoizedFunction = createMemoizedFunction();

在这个例子中,createMemoizedFunction是一个工厂函数,它返回一个具有缓存功能的函数。缓存是通过闭包内部的Map对象实现的。

3. 使用第三方库(如lodash的_.memoize)

如果你不想自己实现缓存逻辑,可以使用像lodash这样的第三方库,它提供了_.memoize函数来轻松创建缓存函数。

const _ = require('lodash');

const memoizedFunction = _.memoize(function(key) {
  return someExpensiveComputation(key);
}, key => key); // 第二个参数是一个解析器函数,用于确定缓存的键

在这个例子中,_.memoize接受两个参数:要缓存的函数和一个解析器函数(用于从原始参数中提取缓存键)。

注意事项

  • 缓存失效:在某些情况下,你可能需要清除或更新缓存。这可以通过在缓存对象上添加额外的方法来实现,如clearCacheupdateCache
  • 内存使用:缓存会占用内存。如果缓存变得太大,可能会导致性能问题。因此,你可能需要实现一种缓存淘汰策略(如LRU - 最近最少使用)来限制缓存的大小。
  • 线程安全:在多线程环境中(如Node.js的worker线程或浏览器中的Web Workers),你需要确保对缓存的访问是线程安全的。这可能需要使用更复杂的同步机制。

函数缓存是一种强大的优化技术,可以显著提高应用程序的性能。然而,它也需要谨慎使用,以避免引入不必要的复杂性和潜在的内存问题。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2221449.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

《汇编语言》第15章——实验15安装新的 int 9 中断例程

安装新的 int9 中断例程 安装一个新的 int 9 中断例程,功能:在 DOS 下,按下A键后,除非不再松开如果松开,就显示满屏幕的A,其他的键照常处理。 提示,按下一个键时产生的扫描码称为通码,松开一个…

云计算作业一hadoop:问题解决备忘

教程地址:https://blog.csdn.net/qq_53877854/article/details/142412784 修改网络配置文件 vim /etc/sysconfig/network-scripts/ifcfg-ens33在root用户下编辑 静态ip地址配置后查看ip与配置不符 注意:确保在这之前已经在VMware的编辑>虚拟网络编…

OpenCV中的图像通道合并

在计算机视觉和图像处理领域,OpenCV是一个强大的工具库,它提供了从基本操作到复杂算法的广泛功能。今天,我们将通过一个简单的示例来探索OpenCV中的图像通道处理,特别是如何操作和理解BGR与RGB颜色空间的差异。 Lena图像&#xf…

WSL迁移到D盘

迁移WSL 下的 ubuntu 到D盘 使用工具LxRunOffline 如果出现下面的错误 使用其他版本 [ERROR] Couldnt set the case sensitive attribute of the directory "\\?\C:\Users\admin\AppData\Local\Packages\CanonicalGroupLimited.Ubuntu20.04LTS_79rhkp1fndgsc\LocalState…

TwinCAT3运动路径生成与执行

右键MAIN,点击Add添加Action,添加如下俩个名称的Action 在MAIN中添加如下代码: 在俩个Action中分别添加如下代码: 在MAIN程序中输入如下代码: 选择激活配置 弹出的对话框选择OK 弹出的对话框选择确定&…

Web前端高级工程师培训:使用 Node.js 构建一个 Web 服务端程序(3)

11、HTTP 协议 11-1、协议的定义 HTTP 是一种能够获取如 HTML 这样的网络资源的 protocol(通讯协议)。它是在 Web 上进行数据交换的基础,是一种 client-server 协议,也就是说,请求通常是由像浏览器这样的接受方发起的。一个完整的Web文档通…

【解决】使用Hypermark将Markdown文件转化为HTML文件

写在前面: 如果文章对你有帮助,记得点赞关注加收藏一波,利于以后需要的时候复习,多谢支持! 文章目录 一、文件准备(一)HTML模板文件(二)MD文件夹和储存文件夹 二、文件转…

【C++贪心】1536. 排布二进制网格的最少交换次数|1880

本文涉及知识点 C贪心 决策包容性 LeetCode1536. 排布二进制网格的最少交换次数 给你一个 n x n 的二进制网格 grid,每一次操作中,你可以选择网格的 相邻两行 进行交换。 一个符合要求的网格需要满足主对角线以上的格子全部都是 0 。 请你返回使网格满…

QUIC 启动!

掘金地址:https://juejin.cn/post/7428200842229006377 引言 QUIC是什么?明明你每天都在用,明明每天都在timing,难道你不知道吗?啊?不会吧,不会吧。 那就让本文来让你全方位的了解这个协议。 …

word表格跨页后自动生成的顶部横线【去除方法】

Hello World! Its been a long time. 这一年重心放在了科研、做事、追寻新的经历上,事有正事、琐事、幸事、哀事,内心与认知成长了一些,思想成熟了几分,技艺也有若干收获。不管怎样,来打个卡吧,纪念一下&…

爬虫日常实战

爬取美团新闻信息,此处采用两种方法实现: 注意点:因为此处的数据都是动态数据,所以一定要考虑好向下滑动数据包会更新的情况,不然就只能读取当前页即第一页数据,方法一通过更新ajax数据包网址页数&#xf…

【MyBatis】初识MyBatis 构建简单框架

目录 MyBatis前言搭建一个简单的MyBatis创建Maven项目引入必要依赖创建数据表结构创建User实体类创建Mapper接口Mapper层Dao层 创建MyBatis的Mapper映射文件编写测试类传统测试类JUnit测试 MyBatis 介绍:MyBatis是一款半自动的ORM持久层框架,具有较高的…

利用自定义 ref 实现函数防抖

今天来简单介绍一个新的方法,使用自定义 ref 实现函数防抖。 1. 自定义 ref 的来源 自定义 ref 防抖函数来自于前端开发中的两个概念:Vue 的响应式系统 和 数防抖(Debounce)。 1、Vue 响应式系统:Vue 提供了 ref 和…

Python学习的自我理解和想法(20)

#1024程序员节|征文# 学的是b站的课程(千锋教育),跟老师写程序,不是自创的代码! 今天是学Python的第20天,学的内容是面向对象中的私有属性,私有方法,多态,单例计模式。开…

【ubuntu18.04】ubuntu18.04升级cmake-3.29.8及还原系统自带cmake操作说明

参考链接 cmake升级、更新(ubuntu18.04)-CSDN博客 升级cmake操作说明 下载链接 Download CMake 下载版本 下载软件包 cmake-3.30.3-linux-x86_64.tar.gz 拷贝软件包到虚拟机 cp /var/run/vmblock-fuse/blockdir/jrY8KS/cmake-3.29.8-linux-x86_64…

spring源码中的,函数式接口,注解@FunctionalInterface

调用方 /org/springframework/beans/factory/support/AbstractBeanFactory.java:333sharedInstance getSingleton(beanName, () -> {try {return createBean(beanName, mbd, args);}catch (BeansException ex) {// Explicitly remove instance from singleton cache: It mi…

高级的SQL查询技巧有哪些?

成长路上不孤单😊😊😊😊😊😊 【14后😊///C爱好者😊///持续分享所学😊///如有需要欢迎收藏转发///😊】 今日分享关于高级SQL查询技巧方面的相关内容&#xf…

MATLAB人脸考勤系统

MATLAB人脸考勤系统课题介绍 该课题为基于MATLAB平台的人脸识别系统。传统的人脸识别都是直接人头的比对,现实意义不大,没有一定的新意。该课题识别原理为:先采集待识别人员的人脸,进行训练,得到人脸特征值。测试的时…

HomeAssistant自定义组件学习-【一】

#环境准备# 按官方的步骤准备就可以,我是在Windows下使用VS Code开发的,安装了WSL(使用模板创建组件需要在WSL环境下完成) 官方链接:https://developers.home-assistant.io/docs/development_environment 环境准备好…

力扣困难题汇总(14道)

题4(困难): 思路: 找两数组中位数,这个看起来简单,顺手反应就是数第(mn)/2个,这个难在要求时间复杂度为log(mn),所以不能这样搞,我的思路是:每次切割长度为较…