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

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

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

相关推荐
一朵梨花压海棠go38 分钟前
html+js实现表格本地筛选
开发语言·javascript·html·ecmascript
蒋星熠43 分钟前
Flutter跨平台工程实践与原理透视:从渲染引擎到高质产物
开发语言·python·算法·flutter·设计模式·性能优化·硬件工程
attitude.x44 分钟前
PyTorch 动态图的灵活性与实用技巧
前端·人工智能·深度学习
β添砖java1 小时前
CSS3核心技术
前端·css·css3
空山新雨(大队长)1 小时前
HTML第八课:HTML4和HTML5的区别
前端·html·html5
小欣加油1 小时前
leetcode 面试题01.02判定是否互为字符重排
数据结构·c++·算法·leetcode·职场和发展
3Cloudream1 小时前
LeetCode 003. 无重复字符的最长子串 - 滑动窗口与哈希表详解
算法·leetcode·字符串·双指针·滑动窗口·哈希表·中等
王璐WL1 小时前
【c++】c++第一课:命名空间
数据结构·c++·算法
猫头虎-前端技术2 小时前
浏览器兼容性问题全解:CSS 前缀、Grid/Flex 布局兼容方案与跨浏览器调试技巧
前端·css·node.js·bootstrap·ecmascript·css3·媒体