# 2023 JavaScript面试题

# 1.new 操作符的工作原理

①.创建一个新的空对象

②.将这个对象的原型设置为函数的 prototype 对象

③.让函数的this指向该对象,为函数添加属性和方法

④.最后返回这个对象

# 2.什么是DOM,什么是BOM?

DOM:文档对象模型,将整个文档看作一个对象,对象中定义了处理网页内容的方法和接口。

BOM:浏览器对象模型,将浏览器看作一个对象,对象中定义了与浏览器进行交互的方法和接口。

🌰常见DOM操作有哪些?

# 3.对类数组对象的理解,如何转换为数组

有length和若干索引属性的对象,与数组类似,但是不能调用数组的方法。

①通过 call 调用函数的 slice、splice 方法

Array.prototype.slice.call(arrayLike, 0);
Array.prototype.splice.call(arrayLike);
1
2

②通过 apply 调用函数的 concat 方法

Array.prototype.concat.apply([], arrayLike);
1

③通过 Array.isArray()

Array.from(arrayLike);
1

🌰arguments 干嘛的?

# 4.对Ajax的理解,如何实现一个Ajax?

通过XMLHttpRequest对象向服务器发送一个异步请求,从服务器拿到数据之后,通过JS操作DOM节点,达到局部刷新的目的。

①首先创建一个XMLHttpRequest对象。

②.open()方法与服务器建立连接,传递请求方式、服务器地址等参数。

③使用onReadystatechange事件,监听服务器端的通信状态。

④最后使用.send()方法发送数据。

🌰服务器端的通信状态有哪些?

# 5.JS为什么会使用变量提升?导致了什么问题?

JS变量提升的目的时为了①提高性能 和 ②增加容错性;

使用变量提升会导致,函数内部变量提升函数内部最顶部,覆盖了外层作用域中传递进来的变量。

👨‍💻写一下

# 6.for...in 和 for...of的区别?🤡

for...in 获取键值,遍历整个原型链,返回数组中所有可枚举对象,性能相对较差;

for...of 获取键名,只遍历对象,返回数组下标对应的属性值。

# 7.JS有哪些数据类型,区别是什么?

原始数据类型:Undefined、Null、Boolean、Number、String;

引用数据类型:数组、对象、函数。

区别在于存储位置不同,原始数据类型占用空间小,大小不固定,存在栈中;引用数据类型占用空间大,大小固定,存在堆中。

BigInt 是一种内置对象,表示任意大的整数,不能与Number一起用,否则就会造成丢失精度的问题,

0.1+0.2=0.3 为什么是 false?

# 8.数据类型检测方法有哪些?

typeof

instanceof

constructor

Object.prototype.toString.call()

# 9.判断数组的方式有哪些?

Object.prototype.toString.call()

_ _ proto _ _

Array.isArray()

instanceof()

Array.prototype.idPrototypeof

# 10.null 和 undefined 的区别?

undefined 是声明了未定义,unll是空对象;

# 11.typeof null = object 为什么?

这是一个历史遗留问题,JS数据类型的底层是以二进制形式存在的,二进制前三位为0 , typeof 就会判定为Object类型,而 null 的二进制恰好都为0,所以 typeof 会误判 null 为 objec t类型。

# 12.Object.is() 与 比较操作符 “===”、“==”的区别

双等号(==):表示两边数据类型不相等,会对类型进行强转再进行比较;

三等号(===):表示两边数据类型不相等,直接返回false;

Object.is()与三等号类似,但是做了一些特殊处理,比如 +0 与 -0 不相等,两个NaN是相等的。

# 13.let、const、var的区别

var 存在提升变量,添加全局属性、重复声明变量、改变指针指向等特点;

const 声明必须设置初始值;

let 可以改变指针方向;

const和let 有块级作用域、存在暂时性死区;

# 14.对闭包的理解

我的理解,闭包就是函数之间的套娃,内层函数可以访问到外层函数的作用域。

如果不是特定任务需要闭包,其他情况下使用闭包对内存和速度的消耗还是蛮大的!

# 15.对作用域、作用域链的理解

作用域:函数和变量能够被访问到的区域。包含全局作用域、函数作用域、块级作用域。

作用域链:当需要使用到某个变量的时候,JS就回去当前作用域中查找,如果找不到,就会去上层作用域继续查找,直到找到为止,这个查找的过程就是一个作用域链。

# 16.对执行上下文的理解

是对JS代码远行环境的抽象概念,只要有js代码远行的地方,就一定运行在执行上下文中。

包含全局执行上下文、函数执行上下文、eval函数执行上下文

# 17.箭头函数(=>)和普通函数(function)有什么区别

①箭头函数更简洁,没有自己的this,继承来的this指向永远不会改变,call、bind、apply也不能改变this指向;

②箭头函数不能做构造函数、generator函数使用;

③没有自己的arguments、prototype;

# 18.箭头函数的this指向哪里?

指向最后一次调用它的对象、指向最后一次调用它的对象、指向最后一次调用它的对象

# 19.对原型、原型链的理解

原型:每个构造函数都有自己的prototype属性,这个属性值是个对象,对象中定义了该函数所有实例共享的属性和方法,这些属性和方法就是原型。

原型链:当需要查找对象的某个属性时,如果对象内部没有这个属性,JS就会去原型对象中查找该属性,直到到达根对象null,找的这个过程就是整个原型链。

# 20.async 和 await 的理解

async 返回一个Promise对象;

await 返回该对象的结果,如果等待的结果不是一个Promise对象,那么就直接返回表达式的结果;如果等待的结果是一个Promise对象,await就会阻塞代码,等待Promise返回value的值,再做await返回结果。

# 21.async await 对比Promsie的优势

①代码阅读起来更加简洁,Promise虽然摆脱了回调地狱,但是then的链式调用看起来很臃肿;

②async await 错误处理更加友好,调试更加方便;

# 22.垃圾回收 与 内存泄露

垃圾回收有引用计数和标记删除两种回收机制

标记删除:变量被声明的时候,就会被加上”存在于上下文“的标记,垃圾回收程序远行的时候,被标记的变量就是待删除变量,随后垃圾回收程序会做一次内存清理,清理所有带标记的变量。

引用计数:变量被引用,引用数+1,变量被覆盖,引用数-1,当变量为0的时候,就会被清理掉回收内存。

# 23.深克隆和浅克隆

# 24.防抖(debounce)与节流(throttle)

防止抖动,节省流量,防抖执行最后一次,节流执行第一次触发的事件。

防抖:固定时间后触发该事件,在固定时间内重复触发,就会重新计时。相当于一个弹簧

节流:固定时间内运行一次,在固定时间内重复触发,只执行之后一次。相当于不间断的水滴

一种优化高频率执行代码的手段

防抖(throttle): n秒后再执行某一事件,若n秒内被重复触发,重新计时

const button = document.querySelector('input');
// 每次事件触发要执行的任务
function payMoney(){
	console.log('点击付款')
}
// 防抖设置
function debounce(){
    func();
}

// 给按钮添加事件监听
button.addEventListener('click',debounce());
1
2
3
4
5
6
7
8
9
10
11
12

引用场景

​ 防抖:百度搜索框,输入猫咪,等待输入完再搜索

​ 节流:

最后更新时间: 6/20/2022, 10:48:50 PM