apply
、call
、bind
三者的区别在于:
this
对象指向this
要指向的对象,如果如果没有这个参数或参数为undefined
或null
,则默认指向全局window
apply
是数组,而call
是参数列表,且apply
和call
是一次性传入参数,而bind
可以分为多次传入bind
是返回绑定this之后的函数,apply
、call
则是立即执行实现bind
的步骤,我们可以分解成为三部分:
this
指向// 方式一:只在bind中传递函数参数
fn.bind(obj,1,2)()
// 方式二:在bind中传递函数参数,也在返回函数中传递参数
fn.bind(obj,1)(2)
new
关键字整体实现代码如下:
复制
Function.prototype.myBind = function (context) {
// 判断调用对象是否为函数
if (typeof this !== "function") {
throw new TypeError("Error");
}
// 获取参数
const args = [...arguments].slice(1),
fn = this;
return function Fn() {
// 根据调用方式,传入不同绑定值
return fn.apply(this instanceof Fn ? new fn(...arguments) : context, args.concat(...arguments));
}
}
具体描述请点击此链接
AJAX
全称(Async Javascript and XML)
即异步的 JavaScript
和 XML
,是一种创建交互式网页应用的网页开发技术,可以在不重新加载整个网页的情况下,与服务器交换数据,并且更新部分网页
Ajax
的原理简单来说通过XmlHttpRequest
对象来向服务器发异步请求,从服务器获得数据,然后用JavaScript
来操作DOM
而更新页面
流程图如下 下面举个例子:领导想找小李汇报一下工作,就委托秘书去叫小李,自己就接着做其他事情,直到秘书告诉他小李已经到了,最后小李跟领导汇报工作
Ajax
请求数据流程与“领导想找小李汇报一下工作”类似,上述秘书就相当于XMLHttpRequest
对象,领导相当于浏览器,响应数据相当于小李.浏览器可以发送HTTP
请求后,接着做其他事情,等收到XHR
返回来的数据再进行操作 实现 Ajax
异步交互需要服务器逻辑进行配合,需要完成以下步骤:
Ajax
的核心对象 XMLHttpRequest
对象XMLHttpRequest
对象的 open()
方法与服务端建立连接 XMLHttpRequest
对象的 send()
方法发送给服务器端XMLHttpRequest
对象提供的 onreadystatechange
事件监听服务器端你的通信状态HTML
页面中具体描述请点击此链接
继承(inheritance)是面向对象软件技术当中的一个概念。如果一个类别B“继承自”另一个类别A,就把这个B称为“A的子类”,而把A称为“B的父类别”也可以称“A是B的超类”。继承可以使得子类具有父类别的各种属性和方法,而不需要再次编写相同的代码。在子类别继承父类别的同时,可以重新定义某些属性,并重写某些方法,即覆盖父类别的原有属性和方法,使其获得与父类别不同的功能。虽然JavaScript
并不是真正的面向对象语言,但它天生的灵活性,使应用场景更加丰富。 JavaScripy
常见的继承方式:
通过Object.create
来划分不同的继承方式,最后的寄生式组合继承方式是通过组合继承改造之后的最优继承方式,而 extends
的语法糖和寄生组合继承的方式基本类似。
具体描述请点击此链接
继承这个概念在面向对象编程思想里面十分重要,也是面试必考的考点之一。javascript的继承主要是依托其原型与原型链的概念来实现的。
ECMAscript将原型链作为实现继承的主要方法。
ES6提供了Class关键字来实现类的定义,Class 可以通过extends关键字实现继承,让子类继承父类的属性和方法。
ES5的四种常用的实现方式:原型链继承、构造函数继承、组合式继承、寄生式组合继承。
具体描述请点击此链接
原型
JavaScript
常被描述为一种基于原型的语言——每个对象拥有一个原型对象
当试图访问一个对象的属性时,它不仅仅在该对象上搜寻,还会搜寻该对象的原型,以及该对象的原型的原型,依次层层向上搜索,直到找到一个名字匹配的属性或到达原型链的末尾。准确地说,这些属性和方法定义在Object的构造器函数(constructor functions)之上的prototype
属性上,而非实例对象本身
下面举个例子:
函数可以有属性。 每个函数都有一个特殊的属性叫作原型prototype
function doSomething(){}
console.log( doSomething.prototype );
控制台输出
{
constructor: ƒ doSomething(),
__proto__: {
constructor: ƒ Object(),
hasOwnProperty: ƒ hasOwnProperty(),
isPrototypeOf: ƒ isPrototypeOf(),
propertyIsEnumerable: ƒ propertyIsEnumerable(),
toLocaleString: ƒ toLocaleString(),
toString: ƒ toString(),
valueOf: ƒ valueOf()
}
}
上面这个对象,就是大家常说的原型对象。可以看到,原型对象有一个自有属性constructor
,这个属性指向该函数,如下图关系展示
原型链
原型对象也可能拥有原型,并从中继承方法和属性,一层一层、以此类推。这种关系常被称为原型链 (prototype chain),它解释了为何一个对象会拥有定义在其他对象中的属性和方法。在对象实例和它的构造器之间建立一个链接(它是__proto__
属性,是从构造函数的prototype
属性派生的),之后通过上溯原型链,在构造器中找到这些属性和方法
下面举个例子:
function Person(name) {
this.name = name;
this.age = 18;
this.sayName = function() {
console.log(this.name);
}
}
// 第二步 创建实例
var person = new Person('person')
下面分析一下:
Person
存在原型对象Person.prototype
person
,person
的__proto__
指向构造函数Person
原型对象Person.prototype.__proto__
指向内置对象,因为 Person.prototype
是个对象,默认是由 Object
函数作为类创建的,而 Object.prototype
为内置对象Person.__proto__
指向内置匿名函数 anonymous
,因为 Person 是个函数对象,默认由 Function 作为类创建Function.prototype
和 Function.__proto__
同时指向内置匿名函数 anonymous
,这样原型链的终点就是 null
总结
下面首先要看几个概念:
__proto__
作为不同对象之间的桥梁,用来指向创建它的构造函数的原型对象的
每个对象的__proto__
都是指向它的构造函数的原型对象prototype
的
person1.__proto__ === Person.prototype
构造函数是一个函数对象,是通过 Function
构造器产生的
Person.__proto__ === Function.prototype
原型对象本身是一个普通对象,而普通对象的构造函数都是Object
Person.prototype.__proto__ === Object.prototype
刚刚上面说了,所有的构造器都是函数对象,函数对象都是 Function
构造产生的
Object.__proto__ === Function.prototype
Object
的原型对象也有__proto__
属性指向null
,null
是原型链的顶端
Object.prototype.__proto__ === null
下面作出总结:
Object
对象,Object
对象直接继承根源对象 null
Object
对象),都是继承自 Function
对象Object
对象直接继承自 Function
对象Function
对象的__proto__
会指向自己的原型对象,最终还是继承自Object
对象先看看在那些场景会导致重复请求:
前端方案
我们可以对症下药:
具体描述请点击此链接
给你一个由 '1'
(陆地)和 '0'
(水)组成的的二维网格,请你计算网格中岛屿的数量。
岛屿总是被水包围,并且每座岛屿只能由水平方向和/或竖直方向上相邻的陆地连接形成。
此外,你可以假设该网格的四条边均被水包围。
具体描述请点击此链接
会继续执行。
虽然Promise是开发过程中使用非常频繁的一个技术点,但是它的一些细节可能很多人都没有去关注过。我们都知道.then
, .catch
, .finally
都可以链式调用,其本质上是因为返回了一个新的Promise实例。
catch的语法形式如下:
1p.catch(onRejected);
.catch
只会处理rejected
的情况,并且也会返回一个新的Promise
实例。
.catch(onRejected)
与then(undefined, onRejected)
在表现上是一致的。
事实上,catch(onRejected)从内部调用了then(undefined, onRejected)。
.catch(onRejected)
的onRejected
回调中返回了一个状态为rejected
的Promise
实例,那么.catch
返回的Promise
实例的状态也将变成rejected
。.catch(onRejected)
的onRejected
回调中抛出了异常,那么.catch
返回的Promise
实例的状态也将变成rejected
。.catch
返回的Promise
实例的状态将是fulfilled
。CORS、Nginx代理跨域、Node中间件代理跨域、WebSocket、postMessage、JSONP
具体描述请点击此链接
懒加载是一种网页性能优化的方式,它能极大的提升用户体验。就比如说图片,图片一直是影响网页性能的主要元凶,现在一张图片超过几兆已经是很经常的事了。如果每次进入页面就请求所有的图片资源,那么可能等图片加载出来用户也早就走了。所以,我们需要懒加载,进入页面的时候,只请求可视区域的图片资源。
总结出来就两个点:
html 实现:实现方式是给 img
标签加上 loading="lazy"
js 实现:原理通过js监听页面的滚动也能实现。
。就比如说图片,图片一直是影响网页性能的主要元凶,现在一张图片超过几兆已经是很经常的事了。如果每次进入页面就请求所有的图片资源,那么可能等图片加载出来用户也早就走了。所以,我们需要懒加载,进入页面的时候,只请求可视区域的图片资源。
总结出来就两个点:
html 实现:实现方式是给 img
标签加上 loading="lazy"
js 实现:原理通过js监听页面的滚动也能实现。
具体描述请点击此链接