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

友情提醒:这篇文章所用到的方法在: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);

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

相关推荐
上单带刀不带妹1 分钟前
手写 Vue 中虚拟 DOM 到真实 DOM 的完整过程
开发语言·前端·javascript·vue.js·前端框架
前端风云志3 分钟前
typescript结构化类型应用两例
javascript
Kaltistss21 分钟前
98.验证二叉搜索树
算法·leetcode·职场和发展
杨进军22 分钟前
React 创建根节点 createRoot
前端·react.js·前端框架
知己如祭25 分钟前
图论基础(DFS、BFS、拓扑排序)
算法
mit6.82433 分钟前
[Cyclone] 哈希算法 | SIMD优化哈希计算 | 大数运算 (Int类)
算法·哈希算法
c++bug36 分钟前
动态规划VS记忆化搜索(2)
算法·动态规划
ModyQyW37 分钟前
用 AI 驱动 wot-design-uni 开发小程序
前端·uni-app
哪 吒38 分钟前
2025B卷 - 华为OD机试七日集训第5期 - 按算法分类,由易到难,循序渐进,玩转OD(Python/JS/C/C++)
python·算法·华为od·华为od机试·2025b卷
说码解字43 分钟前
Kotlin lazy 委托的底层实现原理
前端