数字相加百搭函数——记忆函数

友情提醒:这篇文章所用到的方法在:juejin.cn/post/730453... 这里都有提到

伪数组

在 JavaScript 中,伪数组(Array-like Object)是指具有类数组特性的对象。它们与普通的 JavaScript 数组非常相似,但却不是真正的数组。伪数组具有以下特点:

  1. 拥有 length 属性:伪数组拥有 length 属性,可以通过该属性获取当前元素的个数。
  2. 可以使用下标访问元素:与数组一样,伪数组也可以通过索引来访问元素。
  3. 不具备数组的原型方法:伪数组并不具备数组的原型方法,如 push、pop、slice 等。

常见的伪数组包括 arguments 对象、DOM 元素集合、NodeList 等。例如:

javascriptCopy 复制代码
// arguments 对象是一个伪数组
function test() {
  console.log(arguments.length);
  console.log(arguments[0]);
}
test(1, 2, 3); // 输出:3 1

// DOM 元素集合也是一个伪数组
const divs = document.getElementsByTagName('div');
console.log(divs.length); // 输出 div 元素的个数
console.log(divs[0]); // 输出第一个 div 元素

虽然伪数组不能像数组一样直接调用数组的方法,但它们可以通过 call 或 apply 方法将数组的方法绑定到自身上,从而实现数组的操作。例如:

javascriptCopy 复制代码
// 将数组的 slice 方法绑定到 arguments 对象上
function test() {
  const args = Array.prototype.slice.call(arguments);
  console.log(args.length);
  console.log(args[0]);
}
test(1, 2, 3); // 输出:3 1

总之,伪数组在 JavaScript 中非常常见,需要注意它们的特点和使用方式。

两数相加

以两数相加为例,我们肯定不能写那种很简单的两数相加函数,所以我们限定传递的参数只能是两个:

js 复制代码
function add(a,b){
    if(arguments.length !=2){
       throw new Error('必须传递两个参数')  //超过两个参数就报错 throw Error
    }
    c=a+b
    return c
}
try{
    console.log(add(1,2));
}catch(e){
    console.log(e);
}
console.log('continue....');

优化

摒弃下面那段,我们还可以优化代码。让两数相加传递的参数只允许是整数:

js 复制代码
// 不够优化,得记住之前计算过的
function add(a,b){
    if(arguments.length !== 2 ){
        throw new Error('传递的参数有误');
    }
    if(typeof a!=='number' || typeof b!=='number'){
        throw new Error('必须是整数');
    }
    return a + b
}

记忆函数

我们还可以优化代码。两数相加可以执行很多次,但是如果传递的参数被执行过就会浪费时间,因此我们得让它记住我们之前执行过的参数:

js 复制代码
function add(a,b){
    if(arguments.length !== 2 ){
        throw new Error('传递的参数有误');
    }
    if(typeof a!=='number' || typeof b!=='number'){
        throw new Error('必须是整数');
    }
    return a + b
}

// 记忆函数 百搭
function memorize(f){
  if(typeof f !== 'function')return;
  var cache = {}  // 空间换时间 自由变量
  return function(){
     var key = arguments.length + 
     Array.prototype.join.call(arguments,',')
     if(key in cache){
        return cache[key]

     }else{
        return cache[key] = f.apply(this,arguments)
     }
  }
}
const memorizeAdd = memorize(add);
console.log(memorizeAdd(1,2));
console.log(memorizeAdd(1,2));
// memorizeAdd(1,2)  //传到return function(){}

斐波那契数列

让我们拿记忆函数写个斐波那契数列题目:

js 复制代码
let count = 0
var fibonacci = function(n){
    count++
    return n<2? n:fibonacci(n-1) + fibonacci(n-2);
}

function memorize(f){
    if(typeof f !== 'function')return;
    var cache = {}  // 空间换时间 自由变量
    return function(){
       var key = arguments.length + 
       Array.prototype.join.call(arguments,',')
       if(key in cache){
          return cache[key]
  
       }else{
          return cache[key] = f.apply(this,arguments)
       }
    }
  }
fibonacci = memorize(fibonacci)
fibonacci(20)
console.log(count);

执行时间已经非常小了。这就是记忆函数的可怕与厉害。

相关推荐
Mintopia2 分钟前
⚡当 Next.js 遇上实时通信:Socket.io 与 Pusher 双雄传
前端·后端·全栈
tangdou3690986555 分钟前
可怕!我的Nodejs系统因为日志打印了Error 对象就崩溃了😱 Node.js System Crashed Because of Logging
前端·javascript·后端
刘海东刘海东7 分钟前
结构型智能科技理论研究(草稿)
科技·算法
Takklin8 分钟前
Vue 与 React 应用初始化机制对比 - 前端框架思考笔记
前端·react.js
Mintopia11 分钟前
🎨 数据增强技术在 AIGC 训练中的应用:提升 Web 生成的多样性
前端·javascript·aigc
华仔啊11 分钟前
如何用 Vue3 打造高级音乐播放器?进度条+可视化效果,代码简洁可复用!
前端·css·vue.js
小傅哥12 分钟前
新项目完结,Ai Agent 智能体、拖拉拽编排!
前端·后端
xiaohua0708day14 分钟前
关于解决js中MediaRecorder录制的webm视频没有进度条的问题
javascript·音视频
程序铺子16 分钟前
如何使用 npm 安装 sqlite3 和 canvas 这些包
javascript·npm·node.js
C嘎嘎嵌入式开发18 分钟前
(10)100天python从入门到拿捏《Python中的数据结构与自定义数据结构》
数据结构·python·算法