一文帮你搞定90%的JS手写题
还在害怕手写题吗,本文可以帮你扩展并巩固自己的JS基础,顺便搞定90%的手写题。在工作中还可以对常用的需求进行手写实现,比如深拷贝、防抖节流等可以直接用于往后的项目中,提高项目开发效率。不说废话了,下面就直接上代码吧。1.call的实现
[*]第一个参数为null或者undefined时,this指向全局对象window,值为原始值的指向该原始值的自动包装对象,如 String、Number、Boolean
[*]为了避免函数名与上下文(context)的属性发生冲突,使用Symbol类型作为唯一值
[*]将函数作为传入的上下文(context)属性执行
[*]函数执行完成后删除该属性
[*]返回执行结果
Function.prototype.myCall = function(context,...args){ let cxt = context || window; //将当前被调用的方法定义在cxt.func上.(为了能以对象调用形式绑定this) //新建一个唯一的Symbol变量避免重复 let func = Symbol() cxt = this; args = args ? args : [] //以对象调用形式调用func,此时this指向cxt 也就是传入的需要绑定的this指向 const res = args.length > 0 ? cxt(...args) : cxt(); //删除该方法,不然会对传入对象造成污染(添加该方法) delete cxt; return res;}2.apply的实现
[*]前部分与call一样
[*]第二个参数可以不传,但类型必须为数组或者类数组
Function.prototype.myApply = function(context,args = []){ let cxt = context || window; //将当前被调用的方法定义在cxt.func上.(为了能以对象调用形式绑定this) //新建一个唯一的Symbol变量避免重复 let func = Symbol() cxt = this; //以对象调用形式调用func,此时this指向cxt 也就是传入的需要绑定的this指向 const res = args.length > 0 ? cxt(...args) : cxt(); delete cxt; return res;}3.bind的实现
需要考虑:
[*]bind() 除了 this 外,还可传入多个参数;
[*]bind 创建的新函数可能传入多个参数;
[*]新函数可能被当做构造函数调用;
[*]函数可能有返回值;
实现方法:
[*]bind 方法不会立即执行,需要返回一个待执行的函数;(闭包)
[*]实现作用域绑定(apply)
[*]参数传递(apply 的数组传参)
[*]当作为构造函数的时候,进行原型继承
Function.prototype.myBind = function (context, ...args) { //新建一个变量赋值为this,表示当前函数 const fn = this //判断有没有传参进来,若为空则赋值[] args = args ? args : [] //返回一个newFn函数,在里面调用fn return function newFn(...newFnArgs) { if (this instanceof newFn) { return new fn(...args, ...newFnArgs) } return fn.apply(context, [...args,...newFnArgs]) }}
[*]测试
let name = '小王',age =17;let obj = { name:'小张', age: this.age, myFun: function(from,to){ console.log(this.name + ' 年龄 ' + this.age+'来自 '+from+'去往'+ to) }}let db = { name: '德玛', age: 99}//结果obj.myFun.myCall(db,'成都','上海'); // 德玛 年龄 99 来自 成都去往上海obj.myFun.myApply(db,['成都','上海']); // 德玛 年龄 99 来自 成都去往上海obj.myFun.myBind(db,'成都','上海')(); // 德玛 年龄 99 来自 成都去往上海obj.myFun.myBind(db,['成都','上海'])(); // 德玛 年龄 99 来自 成都, 上海去往 undefined4.寄生式组合继承
function Person(obj) { this.name = obj.name this.age = obj.age}Person.prototype.add = function(value){ console.log(value)}var p1 = new Person({name:"番茄", age: 18})function Person1(obj) { Person.call(this, obj) this.sex = obj.sex}// 这一步是继承的关键Person1.prototype = Object.create(Person.prototype);Person1.prototype.constructor = Person1;Person1.prototype.play = function(value){ console.log(value)}var p2 = new Person1({name:"鸡蛋", age: 118, sex: "男"})5.ES6继承
//class 相当于es5中构造函数//class中定义方法时,前后不能加function,全部定义在class的protopyte属性中//class中定义的所有方法是不可枚举的//class中只能定义方法,不能定义对象,变量等//class和方法内默认都是严格模式//es5中constructor为隐式属性class People{ constructor(name='wang',age='27'){ this.name = name; this.age = age; } eat(){ console.log(`${this.name} ${this.age} eat food`) }}//继承父类class Woman extends People{ constructor(name = 'ren',age = '27'){ //继承父类属性 super(name, age); } eat(){ //继承父类方法 super.eat() } } let wonmanObj=new Woman('xiaoxiami'); wonmanObj.eat();//es5继承先创建子类的实例对象,然后再将父类的方法添加到this上(Parent.apply(this))。 //es6继承是使用关键字super先创建父类的实例对象this,最后在子类class中修改this。6.new的实现
[*]一个继承自 Foo.prototype 的新对象被创建。
[*]使用指定的参数调用构造函数 Foo,并将 this 绑定到新创建的对象。new Foo 等同于 new Foo(),也就是没有指定参数列表,Foo 不带任何参数调用的情况。
[*]由构造函数返回的对象就是 new 表达式的结果。如果构造函数没有显式返回一个对象,则使用步骤1创建的对象。
[*]一般情况下,构造函数不返回值,但是用户可以选择主动返回对象,来覆盖正常的对象创建步骤
function Ctor(){ ....}function myNew(ctor,...args){ if(typeof ctor !== 'function'){ throw 'myNew function the first param must be a function'; } var newObj = Object.create(ctor.prototype); //创建一个继承自ctor.prototype的新对象 var ctorReturnResult = ctor.apply(newObj, args); //将构造函数ctor的this绑定到newObj中 var isObject = typeof ctorReturnResult === 'object' && ctorReturnResult !== null; var isFunction = typeof ctorReturnResult === 'function'; if(isObject || isFunction){ return ctorReturnResult; } return newObj;}let c = myNew(Ctor);7.instanceof的实现
[*]instanceof 是用来判断A是否为B的实例,表达式为:A instanceof B,如果A是B的实例,则返回true,否则返回false。
[*]instanceof 运算符用来测试一个对象在其原型链中是否存在一个构造函数的 prototype 属性。
[*]不能检测基本数据类型,在原型链上的结果未必准确,不能检测null,undefined
[*]实现:遍历左边变量的原型链,直到找到右边变量的 prototype,如果没有找到,返回 false
function myInstanceOf(a,b){ let left = a.__proto__; let right = b.prototype; while(true){ if(left == null){ return false } if(left == right){ return true } left = left.__proto__ }}//instanceof 运算符用于判断构造函数的 prototype 属性是否出现在对象的原型链中的任何位置。function myInstanceof(left, right) { let proto = Object.getPrototypeOf(left), // 获取对象的原型 prototype = right.prototype; // 获取构造函数的 prototype 对象 // 判断构造函数的 prototype 对象是否在对象的原型链上 while (true) { if (!proto) return false; if (proto === prototype) return true; proto = Object.getPrototypeOf(proto); }}8.Object.create()的实现
[*]MDN文档
[*]Object.create()会将参数对象作为一个新创建的空对象的原型, 并返回这个空对象
//简略版function myCreate(obj){ // 新声明一个函数 function C(){}; // 将函数的原型指向obj C.prototype = obj; // 返回这个函数的实力化对象 return new C()}//官方版Polyfillif (typeof Object.create !== "function") { Object.create = function (proto, propertiesObject) { if (typeof proto !== 'object' && typeof proto !== 'function') { throw new TypeError('Object prototype may only be an Object: ' + proto); } else if (proto === null) { throw new Error("This browser's implementation of Object.create is a shim and doesn't support 'null' as the first argument."); } if (typeof propertiesObject !== 'undefined') throw new Error("This browser's implementation of Object.create is a shim and doesn't support a second argument."); function F() {} F.prototype = proto; return new F(); };}9.实现 Object.assign
Object.assign2 = function(target, ...source) { if (target == null) { throw new TypeError('Cannot convert undefined or null to object') } let ret = Object(target) source.forEach(function(obj) { if (obj != null) { for (let key in obj) { if (obj.hasOwnProperty(key)) { ret = obj } } } }) return ret}10.Promise的实现
实现 Promise 需要完全读懂 Promise A+ 规范,不过从总体的实现上看,有如下几个点需要考虑到:
[*]Promise本质是一个状态机,且状态只能为以下三种:Pending(等待态)、Fulfilled(执行态)、Rejected(拒绝态),状态的变更是单向的,只能从Pending -> Fulfilled 或 Pending -> Rejected,状态变更不可逆
[*]then 需要支持链式调用
class Promise { callbacks = []; state = 'pending';//增加状态 value = null;//保存结果 constructor(fn) { fn(this._resolve.bind(this), this._reject.bind(this)); } then(onFulfilled, onRejected) { return new Promise((resolve, reject) => { this._handle({ onFulfilled: onFulfilled || null, onRejected: onRejected || null, resolve: resolve, reject: reject }); }); } _handle(callback) { if (this.state === 'pending') { this.callbacks.push(callback); return; } let cb = this.state === 'fulfilled' ? callback.onFulfilled : callback.onRejected; if (!cb) {//如果then中没有传递任何东西 cb = this.state === 'fulfilled' ? callback.resolve : callback.reject; cb(this.value); return; } let ret = cb(this.value); cb = this.state === 'fulfilled' ? callback.resolve : callback.reject; cb(ret); } _resolve(value) { if (value && (typeof value === 'object' || typeof value === 'function')) { var then = value.then; if (typeof then === 'function') { then.call(value, this._resolve.bind(this), this._reject.bind(this)); return; } } this.state = 'fulfilled';//改变状态 this.value = value;//保存结果 this.callbacks.forEach(callback => this._handle(callback)); } _reject(error) { this.state = 'rejected'; this.value = error; this.callbacks.forEach(callback => this._handle(callback)); }}Promise.resolve
[*]Promsie.resolve(value) 可以将任何值转成值为 value 状态是 fulfilled 的 Promise,但如果传入的值本身是 Promise 则会原样返回它。
Promise.resolve(value) { if (value && value instanceof Promise) { return value; } else if (value && typeof value === 'object' && typeof value.then === 'function') { let then = value.then; return new Promise(resolve => { then(resolve); }); } else if (value) { return new Promise(resolve => resolve(value)); } else { return new Promise(resolve => resolve()); }}Promise.reject
[*]和 Promise.resolve() 类似,Promise.reject() 会实例化一个 rejected 状态的 Promise。但与 Promise.resolve() 不同的是,如果给 Promise.reject() 传递一个 Promise 对象,则这个对象会成为新 Promise 的值。
Promise.reject = function(reason) { return new Promise((resolve, reject) => reject(reason))}Promise.all
[*]传入的所有 Promsie 都是 fulfilled,则返回由他们的值组成的,状态为 fulfilled 的新 Promise;
[*]只要有一个 Promise 是 rejected,则返回 rejected 状态的新 Promsie,且它的值是第一个 rejected 的 Promise 的值;
[*]只要有一个 Promise 是 pending,则返回一个 pending 状态的新 Promise;
Promise.all = function(promiseArr) { let index = 0, result = [] return new Promise((resolve, reject) => { promiseArr.forEach((p, i) => { Promise.resolve(p).then(val => { index++ result = val if (index === promiseArr.length) { resolve(result) } }, err => { reject(err) }) }) })}Promise.race
[*]Promise.race 会返回一个由所有可迭代实例中第一个 fulfilled 或 rejected 的实例包装后的新实例。
Promise.race = function(promiseArr) { return new Promise((resolve, reject) => { promiseArr.forEach(p => { Promise.resolve(p).then(val => { resolve(val) }, err => { rejecte(err) }) }) })}11.Ajax的实现
function ajax(url,method,body,headers){ return new Promise((resolve,reject)=>{ let req = new XMLHttpRequest(); req.open(methods,url); for(let key in headers){ req.setRequestHeader(key,headers) } req.onreadystatechange(()=>{ if(req.readystate == 4){ if(req.status >= '200' && req.status <= 300){ resolve(req.responeText) }else{ reject(req) } } }) req.send(body) })}12.实现防抖函数(debounce)
[*]连续触发在最后一次执行方法,场景:输入框匹配
let debounce = (fn,time = 1000) => { let timeLock = null return function (...args){ clearTimeout(timeLock) timeLock = setTimeout(()=>{ fn(...args) },time) }}13.实现节流函数(throttle)
[*]在一定时间内只触发一次,场景:长列表滚动节流
let throttle = (fn,time = 1000) => { let flag = true; return function (...args){ if(flag){ flag = false; setTimeout(()=>{ flag = true; fn(...args) },time) } }}14.深拷贝(deepclone)
[*]判断类型,正则和日期直接返回新对象
[*]空或者非对象类型,直接返回原值
[*]考虑循环引用,判断如果hash中含有直接返回hash中的值
[*]新建一个相应的new obj.constructor加入hash
[*]遍历对象递归(普通key和key是symbol情况)
function deepClone(obj,hash = new WeakMap()){ if(obj instanceof RegExp) return new RegExp(obj); if(obj instanceof Date) return new Date(obj); if(obj === null || typeof obj !== 'object') return obj; //循环引用的情况 if(hash.has(obj)){ return hash.get(obj) } //new 一个相应的对象 //obj为Array,相当于new Array() //obj为Object,相当于new Object() let constr = new obj.constructor(); hash.set(obj,constr); for(let key in obj){ if(obj.hasOwnProperty(key)){ constr = deepClone(obj,hash) } } //考虑symbol的情况 let symbolObj = Object.getOwnPropertySymbols(obj) for(let i=0;i<symbolObj.length;i++){ if(obj.hasOwnProperty(symbolObj)){ constr] = deepClone(obj],hash) } } return constr}15.数组扁平化的实现(flat)
let arr = ]]]console.log(arr.flat(Infinity))//flat参数为指定要提取嵌套数组的结构深度,默认值为 1//用reduce实现function fn(arr){ return arr.reduce((prev,cur)=>{ return prev.concat(Array.isArray(cur)?fn(cur):cur) },[])}16.函数柯里化
function sumFn(a,b,c){return a+ b + c};let sum = curry(sumFn);sum(2)(3)(5)//10sum(2,3)(5)//10function curry(fn,...args){ let fnLen = fn.length, argsLen = args.length; //对比函数的参数和当前传入参数 //若参数不够就继续递归返回curry //若参数够就调用函数返回相应的值 if(fnLen > argsLen){ return function(...arg2s){ return curry(fn,...args,...arg2s) } }else{ return fn(...args) }}17.使用闭包实现每隔一秒打印 1,2,3,4
for (var i=1; i<=5; i++) { (function (i) { setTimeout(() => console.log(i), 1000*i) })(i)}18.手写一个 jsonp
const jsonp = function (url, data) { return new Promise((resolve, reject) => { // 初始化url let dataString = url.indexOf('?') === -1 ? '?' : '' let callbackName = `jsonpCB_${Date.now()}` url += `${dataString}callback=${callbackName}` if (data) { // 有请求参数,依次添加到url for (let k in data) { url += `${k}=${data}` } } let jsNode = document.createElement('script') jsNode.src = url // 触发callback,触发后删除js标签和绑定在window上的callback window = result => { delete window document.body.removeChild(jsNode) if (result) { resolve(result) } else { reject('没有返回数据') } } // js加载异常的情况 jsNode.addEventListener('error', () => { delete window document.body.removeChild(jsNode) reject('JavaScript资源加载失败') }, false) // 添加js节点到document上时,开始请求 document.body.appendChild(jsNode) })}jsonp('http://192.168.0.103:8081/jsonp', { a: 1, b: 'heiheihei'}).then(result => { console.log(result)}).catch(err => { console.error(err)})19.手写一个观察者模式
class Subject{ constructor(name){ this.name = name this.observers = [] this.state = 'XXXX' } // 被观察者要提供一个接受观察者的方法 attach(observer){ this.observers.push(observer) } // 改变被观察着的状态 setState(newState){ this.state = newState this.observers.forEach(o=>{ o.update(newState) }) }}class Observer{ constructor(name){ this.name = name } update(newState){ console.log(`${this.name}say:${newState}`) }}// 被观察者 灯let sub = new Subject('灯')let mm = new Observer('小明')let jj = new Observer('小健') // 订阅 观察者sub.attach(mm)sub.attach(jj) sub.setState('灯亮了来电了')20.EventEmitter 实现
class EventEmitter { constructor() { this.events = {}; } on(event, callback) { let callbacks = this.events || []; callbacks.push(callback); this.events = callbacks; return this; } off(event, callback) { let callbacks = this.events; this.events = callbacks && callbacks.filter(fn => fn !== callback); return this; } emit(event, ...args) { let callbacks = this.events; callbacks.forEach(fn => { fn(...args); }); return this; } once(event, callback) { let wrapFun = function (...args) { callback(...args); this.off(event, wrapFun); }; this.on(event, wrapFun); return this; }}21.生成随机数的各种方法?
function getRandom(min, max) { return Math.floor(Math.random() * (max - min)) + min }22.如何实现数组的随机排序?
let arr = arr.sort(randomSort)function randomSort(a, b) { return Math.random() > 0.5 ? -1 : 1;}23.写一个通用的事件侦听器函数。
const EventUtils = { // 视能力分别使用dom0||dom2||IE方式 来绑定事件 // 添加事件 addEvent: function(element, type, handler) { if (element.addEventListener) { element.addEventListener(type, handler, false); } else if (element.attachEvent) { element.attachEvent("on" + type, handler); } else { element["on" + type] = handler; } }, // 移除事件 removeEvent: function(element, type, handler) { if (element.removeEventListener) { element.removeEventListener(type, handler, false); } else if (element.detachEvent) { element.detachEvent("on" + type, handler); } else { element["on" + type] = null; } }, // 获取事件目标 getTarget: function(event) { return event.target || event.srcElement; }, // 获取 event 对象的引用,取到事件的所有信息,确保随时能使用 event getEvent: function(event) { return event || window.event; }, // 阻止事件(主要是事件冒泡,因为 IE 不支持事件捕获) stopPropagation: function(event) { if (event.stopPropagation) { event.stopPropagation(); } else { event.cancelBubble = true; } }, // 取消事件的默认行为 preventDefault: function(event) { if (event.preventDefault) { event.preventDefault(); } else { event.returnValue = false; } }};24.使用迭代的方式实现 flatten 函数。
var arr = , ]]]/** * 使用递归的方式处理 * wrap 内保存结果 ret * 返回一个递归函数 **/function wrap() { var ret = []; return function flat(a) { for (var item of a) { if (item.constructor === Array) { ret.concat(flat(item)) } else { ret.push(item) } } return ret }} console.log(wrap()(arr));25.怎么实现一个sleep
[*]sleep函数作用是让线程休眠,等到指定时间在重新唤起。
function sleep(delay) { var start = (new Date()).getTime(); while ((new Date()).getTime() - start < delay) { continue; }}function test() { console.log('111'); sleep(2000); console.log('222');}test()26.实现正则切分千分位(10000 => 10,000)
//无小数点let num1 = '1321434322222'num1.replace(/(\d)(?=(\d{3})+$)/g,'$1,')//有小数点let num2 = '342243242322.3432423'num2.replace(/(\d)(?=(\d{3})+\.)/g,'$1,')27.对象数组去重
输入:[{a:1,b:2,c:3},{b:2,c:3,a:1},{d:2,c:2}]输出:[{a:1,b:2,c:3},{d:2,c:2}]
[*]首先写一个函数把对象中的key排序,然后再转成字符串
[*]遍历数组利用Set将转为字符串后的对象去重
function objSort(obj){ let newObj = {} //遍历对象,并将key进行排序 Object.keys(obj).sort().map(key => { newObj = obj }) //将排序好的数组转成字符串 return JSON.stringify(newObj)}function unique(arr){ let set = new Set(); for(let i=0;i<arr.length;i++){ let str = objSort(arr) set.add(str) } //将数组中的字符串转回对象 arr = [...set].map(item => { return JSON.parse(item) }) return arr}28.解析 URL Params 为对象
let url = 'http://www.domain.com/?user=anonymous&id=123&id=456&city=%E5%8C%97%E4%BA%AC&enabled';parseParam(url)/* 结果{ user: 'anonymous', id: [ 123, 456 ], // 重复出现的 key 要组装成数组,能被转成数字的就转成数字类型 city: '北京', // 中文需解码 enabled: true, // 未指定值得 key 约定为 true}*/function parseParam(url) { const paramsStr = /.+\?(.+)$/.exec(url); // 将 ? 后面的字符串取出来 const paramsArr = paramsStr.split('&'); // 将字符串以 & 分割后存到数组中 let paramsObj = {}; // 将 params 存到对象中 paramsArr.forEach(param => { if (/=/.test(param)) { // 处理有 value 的参数 let = param.split('='); // 分割 key 和 value val = decodeURIComponent(val); // 解码 val = /^\d+$/.test(val) ? parseFloat(val) : val; // 判断是否转为数字 if (paramsObj.hasOwnProperty(key)) { // 如果对象有 key,则添加一个值 paramsObj = [].concat(paramsObj, val); } else { // 如果对象没有这个 key,创建 key 并设置值 paramsObj = val; } } else { // 处理没有 value 的参数 paramsObj = true; } }) return paramsObj;}29.模板引擎实现
let template = '我是{{name}},年龄{{age}},性别{{sex}}';let data = { name: '姓名', age: 18}render(template, data); // 我是姓名,年龄18,性别undefinedfunction render(template, data) { const reg = /\{\{(\w+)\}\}/; // 模板字符串正则 if (reg.test(template)) { // 判断模板里是否有模板字符串 const name = reg.exec(template); // 查找当前模板里第一个模板字符串的字段 template = template.replace(reg, data); // 将第一个模板字符串渲染 return render(template, data); // 递归的渲染并返回渲染后的结构 } return template; // 如果模板没有模板字符串直接返回}30.转化为驼峰命名
var s1 = "get-element-by-id"// 转化为 getElementByIdvar f = function(s) { return s.replace(/-\w/g, function(x) { return x.slice(1).toUpperCase(); })}31.查找字符串中出现最多的字符和个数
[*]例: abbcccddddd -> 字符最多的是d,出现了5次
let str = "abcabcabcbbccccc";let num = 0;let char = ''; // 使其按照一定的次序排列str = str.split('').sort().join('');// "aaabbbbbcccccccc"// 定义正则表达式let re = /(\w)\1+/g;str.replace(re,($0,$1) => { if(num < $0.length){ num = $0.length; char = $1; }});console.log(`字符最多的是${char},出现了${num}次`);32.图片懒加载
let imgList = [...document.querySelectorAll('img')]let length = imgList.lengthconst imgLazyLoad = function() { let count = 0 return (function() { let deleteIndexList = [] imgList.forEach((img, index) => { let rect = img.getBoundingClientRect() if (rect.top < window.innerHeight) { img.src = img.dataset.src deleteIndexList.push(index) count++ if (count === length) { document.removeEventListener('scroll', imgLazyLoad) } } }) imgList = imgList.filter((img, index) => !deleteIndexList.includes(index)) })()}// 这里最好加上防抖处理document.addEventListener('scroll', imgLazyLoad)
参考资料
[*]高频 JavaScript 手写
[*]初、中级前端应该要掌握的手写代码实现
[*]22 道高频 JavaScript 手写
[*]死磕 36 个 JS 手写题(搞懂后,提升真的大)
页:
[1]