debounce(防抖)
防抖可以让多个顺序调用的事件合成一个,防止抖动,防止正常需要一次执行就可以的事件执行了多次。在用户停止触发事件的时候,才进行执行事件。
例如在搜索引擎输入框中输入文字时,每输入一个键盘字符,搜索框会实时将输入值通过请求发送到后台。将每个用户输入的字符都发送至后台,会导致请求过于频繁,造成资源的浪费。
同时用户的体验也不佳,每输入一个字符都发送请求,对于服务器的性能要求很高,因为如果请求耗时太长,就无法做到每敲一个字符,就马上出来关联的输入提示内容,需要很优秀的服务器性能来配合实现。所以一般来说,这样的输入框都需要用防抖来进行处理。除非数据检索或者代码执行耗时足够短,可以不做处理。
同样适用的场景有浏览器的大小调整resize监听,滚动条滚动scroll监听,按钮点击click事件等。

1 2 3 4 5 6 7 8 9 10 11 12
|
const debounce = (func, wait) => { let timeout = null; return () => { clearTimeout(timeout); timeout = setTimeout(func, wait); } }
|
1 2 3
| const testDebounce = (e) => { console.log(e) } document.addEventListener('scroll', debounce(testDebounce, 300));
|
这里为什么要return一个函数,因为timeout变量存在于debounce局部作用域内,正常调用后timeout变量会被内存回收。返回一个函数,引用了timeout变量,形成了闭包,所以在返回的函数内部可以读取到局部timeout变量。
throttle(节流)
节流,也可以理解为限流。让事件在每个间隔的时间里,只执行一次。和防抖不同,节流保证了在x毫秒的事件内,必然触发一次事件。
例如地铁上下班时间限流,每隔5分钟才能进站一波人,这就是限流。防抖和节流,作用区别不大,在应用场景上有一定的重合,关键在于是否需要在一段时间内必须触发一次事件。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
|
const throttle = (func, wait, mustRun) => { let timeout = null; let startTime = new Date(); return (...args) => { const curTime = new Date(); clearTimeout(timeout); if (curTime - startTime >= mustRun) { func.apply(this, args); startTime = curTime; } else { timeout = setTimeout(func, wait); } } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
|
const throttle = (func, mustRun) => { let previous = 0; return (...args) => { const curTime = new Date(); if (curTime - previous >= mustRun) { func.apply(this, args); previous = curTime; } } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
|
const throttle = (func, mustRun) => { let timeout = null; return (...args) => { if (timeout) { return; } timeout = setTimeout(() => { func.apply(this, args); timeout = null; }, mustRun); } }
|
注意
1 2 3
| const test = (e) => { console.log(e) } document.addEventListener('scroll', throttle(test, 300));
|
new(new运算符)
new
运算符创建一个用户定义的对象类型的实例或具有构造函数的内置对象的实例。
语法 new constructor[([arguments])]
new是创建一个新实例,新实例肯定是一个对象。
New操作符实际上经历以下4个步骤:
- 创建一个新对象
- 将构造函数的作用域赋给新对象(this指向新对象)
- 执行构造函数中的代码(为新对象添加属性)
- 返回新对象,如果是引用类型,返回这个引用类型对象,否则返回新创建对象
new 主要来新建实例,要理解new首先来对JS原型链有一定的认识。JS原型prototype在父类里是protoype,在实例里是[[prototype]],可以通过__proto__访问。因此创建一个新对象,同时将新对象的__proto__设置为父类的prototype,实现继承父类。
例如:
1 2 3
| class A {}; const a = new A(); a.__proto__ === A.prototype;
|
1 2 3 4 5 6 7 8 9 10 11 12 13
|
const newObject = (parentObject, ...args) => { const newObj= Object.create(parentObject.prototype); const result = parentObject.apply(newObj, args); return result instanceof Object ? result : newObj; }
|
deepClone (深拷贝)
网上的深拷贝代码一般都有些问题的,因为对于一些特殊的对象没有进行处理,但是一般也不会出现bug,简单的深拷贝有时也能实现功能。
对象有可能出现循环引用。
1 2 3
| const a = { name: 'ben' }; const b = { a }; a.b = b;
|
结构如下:

实现一 JSON.stringify && JSON.parse
通过JSON的两个方法使对象重新构造成新的对象实现深拷贝,它的问题在于会丢弃对象的constructor,也不支持循环引用。同时必须保证处理的对象为能够被json数据结构表示,不符合转换规则会抛出错误。
不推荐这种实现,如果要用,需要配合错误捕获方法来使用。
1 2
| const a = {}; const b = JSON.parse(JSON.stringify(a));
|
实现二 递归
存在问题,支持的特殊对象不是很多,但是还算优雅,解决了循环引用的问题,同时遇到已经引用过的对象,不再重复循环一遍。可以解决循环引用问题,因为遇到相同对象会从cache里直接拿出来返回,并且拿出来的是已经处理过的对象,不进入循环。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| const deepClone = (obj, cache = new WeakMap()) => { if (!obj instanceof Object) return obj; if (cache.get(obj)) return cache.get(obj); if (obj instanceof Function) { return (...arg) => { obj.apply(this, args) } } if (obj instanceof Date) return new Date(obj); if (obj instanceof RegExp) return new RegExp(obj.source, obj.flags);
const res = Array.isArray(obj) ? [] : {}; cache.set(obj, res);
Object.keys(obj).forEach((key) => { if (obj[key] instanceof Object) { res[key] = deepClone(obj[key], cache); } else { res[key] = obj[key]; } });
return res; }
|
测试用例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51
|
const a = { name: 'ben' }; const b = { a }; a.b = b;
const deepClone = (obj, cache = new WeakMap()) => { if (!obj instanceof Object) return obj; if (cache.get(obj)) return cache.get(obj); if (obj instanceof Function) { return (...arg) => { obj.apply(this, args) } } if (obj instanceof Date) return new Date(obj); if (obj instanceof RegExp) return new RegExp(obj.source, obj.flags);
const res = Array.isArray(obj) ? [] : {}; cache.set(obj, res);
Object.keys(obj).forEach((key) => { console.log(key); if (obj[key] instanceof Object) { res[key] = deepClone(obj[key], cache); } else { res[key] = obj[key]; } });
console.log(res, 'res'); return res; }
const c = deepClone(a); console.log(a.b.a.b.a.b === c.b.a.b.a.b);
a.age = 1; c.age = 2;
console.log(a); console.log(c);
|
输出
