【面试篇】寒冬求职季之你必须要懂的原生JS(上)

39次阅读

共计 15876 个字符,预计需要花费 40 分钟才能阅读完成。

互联网寒冬之际,各大公司都缩减了 HC,甚至是采取了“裁员”措施,在这样的大环境之下,想要获得一份更好的工作,必然需要付出更多的努力。
一年前,也许你搞清楚闭包,this,原型链,就能获得认可。但是现在,很显然是不行了。本文梳理出了一些面试中有一定难度的高频原生 JS 问题,部分知识点可能你之前从未关注过,或者看到了,却没有仔细研究,但是它们却非常重要。本文将以真实的面试题的形式来呈现知识点,大家在阅读时,建议不要先看我的答案,而是自己先思考一番。尽管,本文所有的答案,都是我在翻阅各种资料,思考并验证之后,才给出的(绝非复制粘贴而来)。但因水平有限,本人的答案未必是最优的,如果您有更好的答案,欢迎给我留言。
本文篇幅较长,但是满满的都是干货!并且还埋伏了可爱的表情包,希望小伙伴们能够坚持读完。
衷心的祝愿大家都能找到心仪的工作。

1. 原始类型有哪几种?null 是对象吗?原始数据类型和复杂数据类型存储有什么区别?

原始类型有 6 种,分别是 undefined,null,bool,string,number,symbol(ES6 新增)。
虽然 typeof null 返回的值是 object, 但是 null 不是对象,而是基本数据类型的一种。
原始数据类型存储在栈内存,存储的是值。
复杂数据类型存储在堆内存,存储的是地址。当我们把对象赋值给另外一个变量的时候,复制的是地址,指向同一块内存空间,当其中一个对象改变时,另一个对象也会变化。

2. typeof 是否正确判断类型? instanceof 呢?instanceof 的实现原理是什么?
首先 typeof 能够正确的判断基本数据类型,但是除了 null, typeof null 输出的是对象。
但是对象来说,typeof 不能正确的判断其类型,typeof 一个函数可以输出 ‘function’, 而除此之外,输出的全是 object, 这种情况下,我们无法准确的知道对象的类型。
instanceof 可以准确的判断复杂数据类型,但是不能正确判断基本数据类型。(正确判断数据类型请戳:https://github.com/YvetteLau/…
instanceof 是通过原型链判断的,A instanceof B, 在 A 的原型链中层层查找,是否有原型等于 B.__proto__,如果一直找到 A 的原型链的顶端(null; 即 Object.prototype.__proto__), 仍然不等于 B.prototype,那么返回 false,否则返回 true.
instanceof 的实现代码:
// L instanceof R
function instance_of(L, R) {//L 表示左表达式,R 表示右表达式
var O = R.prototype;// 取 R 的显式原型
L = L.__proto__; // 取 L 的隐式原型
while (true) {
if (L === null) // 已经找到顶层
return false;
if (O === L) // 当 O 严格等于 L 时,返回 true
return true;
L = L.__proto__; // 继续向上一层原型链查找
}
}

3. for of , for in 和 forEach,map 的区别。

for…of 循环:具有 iterator 接口,就可以用 for…of 循环遍历它的成员(属性值)。for…of 循环可以使用的范围包括数组、Set 和 Map 结构、某些类似数组的对象、Generator 对象,以及字符串。for…of 循环调用遍历器接口,数组的遍历器接口只返回具有数字索引的属性。对于普通的对象,for…of 结构不能直接使用,会报错,必须部署了 Iterator 接口后才能使用。可以中断循环。
for…in 循环:遍历对象自身的和继承的可枚举的属性, 不能直接获取属性值。可以中断循环。
forEach: 只能遍历数组,不能中断,没有返回值(或认为返回值是 undefined),不修改原数组。
map: 只能遍历数组,不能中断,返回值是修改后的数组,不修改原数组。

PS: Object.keys():返回给定对象所有可枚举属性的字符串数组。
如还不了解 iterator 接口或 for…of, 请先阅读 ES6 文档: Iterator 和 for…of 循环
更多细节请戳: https://github.com/YvetteLau/…

4. 如何判断一个变量是不是数组?

使用 Array.isArray 判断,如果返回 true, 说明是数组
使用 instanceof Array 判断,如果返回 true, 说明是数组
使用 Object.prototype.toString.call 判断,如果值是 [object Array], 说明是数组
通过 constructor 来判断,如果是数组,那么 arr.constructor === Array. (不准确,因为我们可以指定 obj.constructor = Array)

function fn() {
console.log(Array.isArray(arguments)); //false; 因为 arguments 是类数组,但不是数组
console.log(Array.isArray([1,2,3,4])); //true
console.log(arguments instanceof Array); //fasle
console.log([1,2,3,4] instanceof Array); //true
console.log(Object.prototype.toString.call(arguments)); //[object Arguments]
console.log(Object.prototype.toString.call([1,2,3,4])); //[object Array]
console.log(arguments.constructor === Array); //false
arguments.constructor = Array;
console.log(arguments.constructor === Array); //true
console.log(Array.isArray(arguments)); //false
}
fn(1,2,3,4);

5. 类数组和数组的区别是什么?
类数组:
1)拥有 length 属性,其它属性(索引)为非负整数(对象中的索引会被当做字符串来处理);
2)不具有数组所具有的方法;
类数组是一个普通对象,而真实的数组是 Array 类型。
常见的类数组有: 函数的参数 arugments, DOM 对象列表(比如通过 document.querySelectorAll 得到的列表), jQuery 对象 (比如 $(“div”)).
类数组可以转换为数组:
// 第一种方法
Array.prototype.slice.call(arrayLike);
// 第二种方法
[…arrayLike];
// 第三种方法:
Array.from(arrayLike);
PS: 任何定义了遍历器(Iterator)接口的对象,都可以用扩展运算符转为真正的数组。
Array.from 方法用于将两类对象转为真正的数组:类似数组的对象(array-like object)和可遍历(iterable)的对象。
6. == 和 === 有什么区别?
=== 不需要进行类型转换,只有类型相同并且值相等时,才返回 true.
== 如果两者类型不同,首先需要进行类型转换。具体流程如下:

首先判断两者类型是否相同,如果相等,判断值是否相等.
如果类型不同,进行类型转换
判断比较的是否是 null 或者是 undefined, 如果是, 返回 true .
判断两者类型是否为 string 和 number, 如果是, 将字符串转换成 number
判断其中一方是否为 boolean, 如果是, 将 boolean 转为 number 再进行判断
判断其中一方是否为 object 且另一方为 string、number 或者 symbol , 如果是, 将 object 转为原始类型再进行判断

let person1 = {
age: 25
}
let person2 = person1;
person2.gae = 20;
console.log(person1 === person2); //true, 注意复杂数据类型,比较的是引用地址
思考: [] == ![]

我们来分析一下: [] == ![] 是 true 还是 false?

首先,我们需要知道 ! 优先级是高于 == (更多运算符优先级可查看: 运算符优先级)

![] 引用类型转换成布尔值都是 true, 因此![]的是 false
根据上面的比较步骤中的第五条,其中一方是 boolean,将 boolean 转为 number 再进行判断,false 转换成 number,对应的值是 0.
根据上面比较步骤中的第六条,有一方是 number,那么将 object 也转换成 Number, 空数组转换成数字,对应的值是 0.(空数组转换成数字,对应的值是 0,如果数组中只有一个数字,那么转成 number 就是这个数字,其它情况,均为 NaN)
0 == 0; 为 true

7. ES6 中的 class 和 ES5 的类有什么区别?

ES6 class 内部所有定义的方法都是不可枚举的;
ES6 class 必须使用 new 调用;
ES6 class 不存在变量提升;
ES6 class 默认即是严格模式;
ES6 class 子类必须在父类的构造函数中调用 super(),这样才有 this 对象;ES5 中类继承的关系是相反的,先有子类的 this,然后用父类的方法应用在 this 上。

8. 数组的哪些 API 会改变原数组?
修改原数组的 API 有:
splice/reverse/fill/copyWithin/sort/push/pop/unshift/shift
不修改原数组的 API 有:
slice/map/forEach/every/filter/reduce/entry/entries/find

9. let、const 以及 var 的区别是什么?

let 和 const 定义的变量不会出现变量提升,而 var 定义的变量会提升。
let 和 const 是 JS 中的块级作用域
let 和 const 不允许重复声明(会抛出错误)
let 和 const 定义的变量在定义语句之前,如果使用会抛出错误(形成了暂时性死区),而 var 不会。
const 声明一个只读的常量。一旦声明,常量的值就不能改变(如果声明是一个对象,那么不能改变的是对象的引用地址)

10. 在 JS 中什么是变量提升?什么是暂时性死区?
变量提升就是变量在声明之前就可以使用,值为 undefined。
在代码块内,使用 let/const 命令声明变量之前,该变量都是不可用的(会抛出错误)。这在语法上,称为“暂时性死区”。暂时性死区也意味着 typeof 不再是一个百分百安全的操作。
typeof x; // ReferenceError(暂时性死区,抛错)
let x;
typeof y; // 值是 undefined, 不会报错
暂时性死区的本质就是,只要一进入当前作用域,所要使用的变量就已经存在了,但是不可获取,只有等到声明变量的那一行代码出现,才可以获取和使用该变量。
11. 如何正确的判断 this? 箭头函数的 this 是什么?
this 的绑定规则有四种:默认绑定,隐式绑定,显式绑定,new 绑定.

函数是否在 new 中调用(new 绑定),如果是,那么 this 绑定的是新创建的对象。
函数是否通过 call,apply 调用,或者使用了 bind (即硬绑定),如果是,那么 this 绑定的就是指定的对象。
函数是否在某个上下文对象中调用(隐式绑定),如果是的话,this 绑定的是那个上下文对象。一般是 obj.foo()
如果以上都不是,那么使用默认绑定。如果在严格模式下,则绑定到 undefined,否则绑定到全局对象。
如果把 null 或者 undefined 作为 this 的绑定对象传入 call、apply 或者 bind, 这些值在调用时会被忽略,实际应用的是默认绑定规则。
箭头函数没有自己的 this, 它的 this 继承于上一层代码块的 this。

测试下是否已经成功 Get 了此知识点(浏览器执行环境):
var number = 5;
var obj = {
number: 3,
fn1: (function () {
var number;
this.number *= 2;
number = number * 2;
number = 3;
return function () {
var num = this.number;
this.number *= 2;
console.log(num);
number *= 3;
console.log(number);
}
})();
}
var fn1 = obj.fn1;
fn1.call(null);
obj.fn1();
console.log(window.number);
如果 this 的知识点,您还不太懂,请戳: 嗨,你真的懂 this 吗?

12. 词法作用域和 this 的区别。

词法作用域是由你在写代码时将变量和块作用域写在哪里来决定的
this 是在调用时被绑定的,this 指向什么,完全取决于函数的调用位置(关于 this 的指向问题,本文已经有说明)

13. 谈谈你对 JS 执行上下文栈和作用域链的理解。
执行上下文就是当前 JavaScript 代码被解析和执行时所在环境, JS 执行上下文栈可以认为是一个存储函数调用的栈结构,遵循先进后出的原则。

JavaScript 执行在单线程上,所有的代码都是排队执行。
一开始浏览器执行全局的代码时,首先创建全局的执行上下文,压入执行栈的顶部。
每当进入一个函数的执行就会创建函数的执行上下文,并且把它压入执行栈的顶部。当前函数执行 - 完成后,当前函数的执行上下文出栈,并等待垃圾回收。
浏览器的 JS 执行引擎总是访问栈顶的执行上下文。
全局上下文只有唯一的一个,它在浏览器关闭时出栈。

作用域链: 无论是 LHS 还是 RHS 查询,都会在当前的作用域开始查找,如果没有找到,就会向上级作用域继续查找目标标识符,每次上升一个作用域,一直到全局作用域为止。

题难不难?不难!继续挑战一下难!知道难,就更要继续了!
14. 什么是闭包?闭包的作用是什么?闭包有哪些使用场景?
闭包是指有权访问另一个函数作用域中的变量的函数,创建闭包最常用的方式就是在一个函数内部创建另一个函数。
闭包的作用有:

封装私有变量
模仿块级作用域(ES5 中没有块级作用域)
实现 JS 的模块

15. call、apply 有什么区别?call,aplly 和 bind 的内部是如何实现的?
call 和 apply 的功能相同,区别在于传参的方式不一样:

fn.call(obj, arg1, arg2, …), 调用一个函数, 具有一个指定的 this 值和分别地提供的参数(参数的列表)。
fn.apply(obj, [argsArray]), 调用一个函数,具有一个指定的 this 值,以及作为一个数组(或类数组对象)提供的参数。

call 核心:

将函数设为传入参数的属性
指定 this 到函数并传入给定参数执行函数
如果不传入参数或者参数为 null,默认指向为 window / global
删除参数上的函数

Function.prototype.call = function (context) {
/** 如果第一个参数传入的是 null 或者是 undefined, 那么指向 this 指向 window/global */
/** 如果第一个参数传入的不是 null 或者是 undefined, 那么必须是一个对象 */
if (!context) {
//context 为 null 或者是 undefined
context = typeof window === ‘undefined’ ? global : window;
}
context.fn = this; //this 指向的是当前的函数(Function 的实例)
let args = […arguments].slice(1);// 获取除了 this 指向对象以外的参数, 空数组 slice 后返回的仍然是空数组
let result = context.fn(…args); // 隐式绑定, 当前函数的 this 指向了 context.
delete context.fn;
return result;
}

// 测试代码
var foo = {
name: ‘Selina’
}
var name = ‘Chirs’;
function bar(job, age) {
console.log(this.name);
console.log(job, age);
}
bar.call(foo, ‘programmer’, 20);
// Selina programmer 20
bar.call(null, ‘teacher’, 25);
// 浏览器环境: Chirs teacher 25; node 环境: undefined teacher 25

apply:
apply 的实现和 call 很类似,但是需要注意他们的参数是不一样的,apply 的第二个参数是数组或类数组.
Function.prototype.apply = function (context, rest) {
if (!context) {
//context 为 null 或者是 undefined 时, 设置默认值
context = typeof window === ‘undefined’ ? global : window;
}
context.fn = this;
let result = context.fn(…rest);
delete context.fn;
return result;
}
var foo = {
name: ‘Selina’
}
var name = ‘Chirs’;
function bar(job, age) {
console.log(this.name);
console.log(job, age);
}
bar.apply(foo, [‘programmer’, 20]);
// Selina programmer 20
bar.apply(null, [‘teacher’, 25]);
// 浏览器环境: Chirs programmer 20; node 环境: undefined teacher 25
bind
bind 和 call/apply 有一个很重要的区别,一个函数被 call/apply 的时候,会直接调用,但是 bind 会创建一个新函数。当这个新函数被调用时,bind() 的第一个参数将作为它运行时的 this,之后的一序列参数将会在传递的实参前传入作为它的参数。
Function.prototype.my_bind = function(context) {
if(typeof this !== “function”){
throw new TypeError(“not a function”);
}
let self = this;
let args = […arguments].slice(1);
function Fn() {};
Fn.prototype = this.prototype;
let bound = function() {
let res = […args, …arguments]; //bind 传递的参数和函数调用时传递的参数拼接
context = this instanceof Fn ? this : context || this;
return self.apply(context, res);
}
// 原型链
bound.prototype = new Fn();
return bound;
}

var name = ‘Jack’;
function person(age, job, gender){
console.log(this.name , age, job, gender);
}
var Yve = {name : ‘Yvette’};
let result = person.my_bind(Yve, 22, ‘enginner’)(‘female’);

16. new 的原理是什么?通过 new 的方式创建对象和通过字面量创建有什么区别?
new:

创建一个新对象。
这个新对象会被执行 [[原型]] 连接。
将构造函数的作用域赋值给新对象,即 this 指向这个新对象.
如果函数没有返回其他对象,那么 new 表达式中的函数调用会自动返回这个新对象。

function new(func) {
lat target = {};
target.__proto__ = func.prototype;
let res = func.call(target);
if (typeof(res) == “object” || typeof(res) == “function”) {
return res;
}
return target;
}
字面量创建对象,不会调用 Object 构造函数, 简洁且性能更好;
new Object() 方式创建对象本质上是方法调用,涉及到在 proto 链中遍历该方法,当找到该方法后,又会生产方法调用必须的 堆栈信息,方法调用结束后,还要释放该堆栈,性能不如字面量的方式。
通过对象字面量定义对象时,不会调用 Object 构造函数。

17. 谈谈你对原型的理解?
在 JavaScript 中,每当定义一个对象(函数也是对象)时候,对象中都会包含一些预定义的属性。其中每个函数对象都有一个 prototype 属性,这个属性指向函数的原型对象。使用原型对象的好处是所有对象实例共享它所包含的属性和方法。

18. 什么是原型链?【原型链解决的是什么问题?】
原型链解决的主要是继承问题。
每个对象拥有一个原型对象,通过 proto (读音: dunder proto) 指针指向其原型对象,并从中继承方法和属性,同时原型对象也可能拥有原型,这样一层一层,最终指向 null(Object.proptotype.__proto__ 指向的是 null)。这种关系被称为原型链 (prototype chain),通过原型链一个对象可以拥有定义在其他对象中的属性和方法。
构造函数 Parent、Parent.prototype 和 实例 p 的关系如下:(p.__proto__ === Parent.prototype)

19. prototype 和 __proto__ 区别是什么?
prototype 是构造函数的属性。
__proto__ 是每个实例都有的属性,可以访问 [[prototype]] 属性。
实例的__proto__ 与其构造函数的 prototype 指向的是同一个对象。
function Student(name) {
this.name = name;
}
Student.prototype.setAge = function(){
this.age=20;
}
let Jack = new Student(‘jack’);
console.log(Jack.__proto__);
//console.log(Object.getPrototypeOf(Jack));;
console.log(Student.prototype);
console.log(Jack.__proto__ === Student.prototype);//true

20. 使用 ES5 实现一个继承?
组合继承(最常用的继承方式)
function SuperType() {
this.name = name;
this.colors = [‘red’, ‘blue’, ‘green’];
}
SuperType.prototype.sayName = function() {
console.log(this.name);
}

function SubType(name, age) {
SuperType.call(this, name);
this.age = age;
}
SubType.prototype = new SuperType();
SubType.prototype.constructor = SubType;

SubType.prototype.sayAge = function() {
console.log(this.age);
}
其它继承方式实现,可以参考《JavaScript 高级程序设计》

21. 什么是深拷贝?深拷贝和浅拷贝有什么区别?
浅拷贝是指只复制第一层对象,但是当对象的属性是引用类型时,实质复制的是其引用,当引用指向的值改变时也会跟着变化。
深拷贝复制变量值,对于非基本类型的变量,则递归至基本类型变量后,再复制。深拷贝后的对象与原来的对象是完全隔离的,互不影响,对一个对象的修改并不会影响另一个对象。
实现一个深拷贝:
function deepClone(obj) {// 递归拷贝
if(obj == null) return null;
if(obj instanceof Date || obj instanceof RegExp) {
return new Date(obj);
}
if(typeof obj !== ‘object’) {
return obj;
}
let t = new obj.constructor();
for(let key in obj) {
t[key] = deepClone(obj[key]);
}
return t;
}

看不下去了?别人的送分题会成为你的送命题
22. 防抖和节流的区别是什么?防抖和节流的实现。
防抖和节流的作用都是防止函数多次调用。区别在于,假设一个用户一直触发这个函数,且每次触发函数的间隔小于设置的时间,防抖的情况下只会调用一次,而节流的情况会每隔一定时间调用一次函数。
防抖(debounce): n 秒内函数只会执行一次,如果 n 秒内高频事件再次被触发,则重新计算时间
function debounce(func, wait, immediate=true) {
let timeout, context, args;
// 延迟执行函数
const later = () => setTimeout(() => {
// 延迟函数执行完毕,清空定时器
timeout = null
// 延迟执行的情况下,函数会在延迟函数中执行
// 使用到之前缓存的参数和上下文
if (!immediate) {
func.apply(context, args);
context = args = null;
}
}, wait);
let debounced = function (…params) {
if (!timeout) {
timeout = later();
if (immediate) {
// 立即执行
func.apply(this, params);
} else {
// 闭包
context = this;
args = params;
}
} else {
clearTimeout(timeout);
timeout = later();
}
}
debounced.cancel = function () {
clearTimeout(timeout);
timeout = null;
};
return debounced;
};
防抖的应用场景:

每次 resize/scroll 触发统计事件
文本输入的验证(连续输入文字后发送 AJAX 请求进行验证,验证一次就好)

节流(throttle): 高频事件在规定时间内只会执行一次,执行一次后,只有大于设定的执行周期后才会执行第二次。
//underscore.js
function throttle(func, wait, options) {
var timeout, context, args, result;
var previous = 0;
if (!options) options = {};

var later = function () {
previous = options.leading === false ? 0 : Date.now() || new Date().getTime();
timeout = null;
result = func.apply(context, args);
if (!timeout) context = args = null;
};

var throttled = function () {
var now = Date.now() || new Date().getTime();
if (!previous && options.leading === false) previous = now;
var remaining = wait – (now – previous);
context = this;
args = arguments;
if (remaining <= 0 || remaining > wait) {
if (timeout) {
clearTimeout(timeout);
timeout = null;
}
previous = now;
result = func.apply(context, args);
if (!timeout) context = args = null;
} else if (!timeout && options.trailing !== false) {
// 判断是否设置了定时器和 trailing
timeout = setTimeout(later, remaining);
}
return result;
};

throttled.cancel = function () {
clearTimeout(timeout);
previous = 0;
timeout = context = args = null;
};

return throttled;
};

函数节流的应用场景有:

DOM 元素的拖拽功能实现(mousemove)
射击游戏的 mousedown/keydown 事件(单位时间只能发射一颗子弹)
计算鼠标移动的距离(mousemove)
Canvas 模拟画板功能(mousemove)
搜索联想(keyup)
监听滚动事件判断是否到页面底部自动加载更多:给 scroll 加了 debounce 后,只有用户停止滚动后,才会判断是否到了页面底部;如果是 throttle 的话,只要页面滚动就会间隔一段时间判断一次

23. 取数组的最大值(ES5、ES6)
// ES5 的写法
Math.max.apply(null, [14, 3, 77, 30]);

// ES6 的写法
Math.max(…[14, 3, 77, 30]);

// reduce
[14,3,77,30].reduce((accumulator, currentValue)=>{
return accumulator = accumulator > currentValue ? accumulator : currentValue
});

24. ES6 新的特性有哪些?

新增了块级作用域(let,const)
提供了定义类的语法糖(class)
新增了一种基本数据类型(Symbol)
新增了变量的解构赋值
函数参数允许设置默认值,引入了 rest 参数,新增了箭头函数
数组新增了一些 API,如 isArray / from / of 方法; 数组实例新增了 entries(),keys() 和 values() 等方法
对象和数组新增了扩展运算符
ES6 新增了模块化(import/export)
ES6 新增了 Set 和 Map 数据结构
ES6 原生提供 Proxy 构造函数,用来生成 Proxy 实例
ES6 新增了生成器 (Generator) 和遍历器(Iterator)

25. setTimeout 倒计时为什么会出现误差?
setTimeout() 只是将事件插入了“任务队列”,必须等当前代码(执行栈)执行完,主线程才会去执行它指定的回调函数。要是当前代码消耗时间很长,也有可能要等很久,所以并没办法保证回调函数一定会在 setTimeout() 指定的时间执行。所以,setTimeout() 的第二个参数表示的是最少时间,并非是确切时间。
HTML5 标准规定了 setTimeout() 的第二个参数的最小值不得小于 4 毫秒,如果低于这个值,则默认是 4 毫秒。在此之前。老版本的浏览器都将最短时间设为 10 毫秒。另外,对于那些 DOM 的变动(尤其是涉及页面重新渲染的部分),通常是间隔 16 毫秒执行。这时使用 requestAnimationFrame() 的效果要好于 setTimeout();

26. 为什么 0.1 + 0.2 != 0.3 ?
0.1 + 0.2 != 0.3 是因为在进制转换和进阶运算的过程中出现精度损失。
下面是详细解释:
JavaScript 使用 Number 类型表示数字(整数和浮点数),使用 64 位表示一个数字。

图片说明:

第 0 位:符号位,0 表示正数,1 表示负数(s)
第 1 位到第 11 位:储存指数部分(e)
第 12 位到第 63 位:储存小数部分(即有效数字)f

计算机无法直接对十进制的数字进行运算, 需要先对照 IEEE 754 规范转换成二进制,然后对阶运算。
1. 进制转换
0.1 和 0.2 转换成二进制后会无限循环
0.1 -> 0.0001100110011001…(无限循环)
0.2 -> 0.0011001100110011…(无限循环)
但是由于 IEEE 754 尾数位数限制,需要将后面多余的位截掉,这样在进制之间的转换中精度已经损失。
2. 对阶运算
由于指数位数不相同,运算时需要对阶运算 这部分也可能产生精度损失。
按照上面两步运算(包括两步的精度损失),最后的结果是
0.0100110011001100110011001100110011001100110011001100
结果转换成十进制之后就是 0.30000000000000004。
27. promise 有几种状态, Promise 有什么优缺点 ?
promise 有三种状态: fulfilled, rejected, resolved.
Promise 的优点:

一旦状态改变,就不会再变,任何时候都可以得到这个结果
可以将异步操作以同步操作的流程表达出来,避免了层层嵌套的回调函数

Promise 的缺点:

无法取消 Promise
当处于 pending 状态时,无法得知目前进展到哪一个阶段

28. Promise 构造函数是同步还是异步执行,then 呢 ?promise 如何实现 then 处理 ?
Promise 的构造函数是同步执行的。then 是异步执行的。
promise 的 then 实现,详见: Promise 源码实现

29. Promise 和 setTimeout 的区别 ?
Promise 是微任务,setTimeout 是宏任务,同一个事件循环中,promise.then 总是先于 setTimeout 执行。同一个事件循环中,promise.then 先于 setTimeout 执行。

30. 如何实现 Promise.all ?
要实现 Promise.all, 首先我们需要知道 Promise.all 的功能:

如果传入的参数是一个空的可迭代对象,那么此 promise 对象回调完成(resolve), 只有此情况,是同步执行的,其它都是异步返回的。
如果传入的参数不包含任何 promise,则返回一个异步完成.

promises 中所有的 promise 都 promise 都“完成”时或参数中不包含 promise 时回调完成。

如果参数中有一个 promise 失败,那么 Promise.all 返回的 promise 对象失败
在任何情况下,Promise.all 返回的 promise 的完成状态的结果都是一个数组

Promise.all = function (promises) {
return new Promise((resolve, reject) => {
let index = 0;
let result = [];
if (promises.length === 0) {
resolve(result);
} else {
setTimeout(() => {
function processValue(i, data) {
result[i] = data;
if (++index === promises.length) {
resolve(result);
}
}
for (let i = 0; i < promises.length; i++) {
//promises[i] 可能是普通值
Promise.resolve(promises[i]).then((data) => {
processValue(i, data);
}, (err) => {
reject(err);
return;
});
}
})
}
});
}
如果想了解更多 Promise 的源码实现,可以参考我的另一篇文章:Promise 的源码实现(完美符合 Promise/A+ 规范)

31. 如何实现 Promise.finnally ?
不管成功还是失败,都会走到 finally 中, 并且 finally 之后,还可以继续 then。并且会将值原封不动的传递给后面的 then.
Promise.prototype.finally = function (callback) {
return this.then((value) => {
return Promise.resolve(callback()).then(() => {
return value;
});
}, (err) => {
return Promise.resolve(callback()).then(() => {
throw err;
});
});
}

32. 什么是函数柯里化?实现 sum(1)(2)(3) 返回结果是 1,2,3 之和
函数柯里化是把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数,并且返回接受余下的参数而且返回结果的新函数的技术。
function sum(a) {
return function(b) {
return function(c) {
return a+b+c;
}
}
}
console.log(sum(1)(2)(3)); // 6
引申:实现一个 curry 函数,将普通函数进行柯里化:
function curry(fn, args = []) {
return function(){
let rest = […args, …arguments];
if (rest.length < fn.length) {
return curry.call(this,fn,rest);
}else{
return fn.apply(this,rest);
}
}
}
//test
function sum(a,b,c) {
return a+b+c;
}
let sumFn = curry(sum);
console.log(sumFn(1)(2)(3)); //6
console.log(sumFn(1)(2, 3)); //6

如果您在面试中遇到了更多的原生 JS 问题,或者有一些本文未涉及到且有一定难度的 JS 知识,请给我留言。您的问题将会出现在后续文章中~

本文的写成耗费了非常多的时间,在这个过程中,我也学习到了很多知识,谢谢各位小伙伴愿意花费宝贵的时间阅读本文,如果本文给了您一点帮助或者是启发,请不要吝啬你的赞和 Star,您的肯定是我前进的最大动力。https://github.com/YvetteLau/…
后续写作计划
1.《寒冬求职季之你必须要懂的原生 JS》(中)(下)
2.《寒冬求职季之你必须要知道的 CSS》
3.《寒冬求职季之你必须要懂的前端安全》
4.《寒冬求职季之你必须要懂的一些浏览器知识》
5.《寒冬求职季之你必须要知道的性能优化》
针对 React 技术栈:
1.《寒冬求职季之你必须要懂的 React》系列
2.《寒冬求职季之你必须要懂的 ReactNative》系列
欢迎关注本人微信公众号:进击的前端小姐姐
参考文章:

https://www.ibm.com/developer…
https://juejin.im/post/5c7736…
选用了面试之道上的部分面试题
选用了木易杨说文中提及的部分面试题: https://juejin.im/post/5bc92e…

特别说明: 0.1 + 0.2 !== 0.3 此题答案大量使用了此篇文章的图文: https://juejin.im/post/5b90e0…

选用了朋友面试大厂时遇到的一些面试题
《你不知道的 JavaSctipt》
《JavaScript 高级程序设计》
https://github.com/hanzichi/u…

正文完
 0