前端高频面试题2:JavaScript/TypeScript

article/2025/6/27 0:27:07

1.什么是类数组对象

        一个拥有 length 属性和若干索引属性的对象就可以被称为类数组对象,类数组对象和数组类似,但是不能调用数组的方法。常见的类数组对象有 arguments 和 DOM 方法的返回结果,还有一个函数也可以被看作是类数组对象,因为它含有 length 属性值,代表可接收的参数个数。

常见的类数组转换为数组的方法有这样几种:

(1)通过 call 调用数组的 slice 方法来实现转换

Array.prototype.slice.call(arrayLike);

(2)通过 call 调用数组的 splice 方法来实现转换

Array.prototype.splice.call(arrayLike, 0);

(3)通过 apply 调用数组的 concat 方法来实现转换

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

(4)通过 Array.from 方法来实现转换

Array.from(arrayLike);

2.Js有哪些基本数据类型

  • 简单类型

    • Number

    • String

    • Boolean

    • Symbol

    • Null

    • Undefined

  • 复杂数据类型

    • Object

    • Array

    • Function

    • Map

    • Set

3.Js判断类型的几种方式,有什么区别

方法支持类型跨框架可靠性原始类型引用类型细分特殊说明
typeof原始类型、函数可靠✔️null 返回 "object"
instanceof引用类型不可靠✔️依赖原型链
Object.prototype.toString所有类型可靠✔️✔️最全面

1.typeof

作用:返回变量的原始类型(字符串形式)。 语法:typeof variable 返回值:

  • "undefined"(未定义)

  • "boolean"(布尔值)

  • "string"(字符串)

  • "number"(数字)

  • "bigint"(BigInt类型)

  • "symbol"(Symbol类型)

  • "function"(函数)

  • "object"(对象、数组、null等)

  • "object"(未识别的ES6+类型,如MapSet

特点:

  • 对原始类型(除 null)有效,但对引用类型无法细分(如数组、对象、null 均返回 "object")。

  • typeof null === "object"(历史遗留问题)。

  • 无法区分对象的具体类型(如 DateRegExp)。

适用场景:快速判断原始类型或函数。

2.instanceof

作用:检测变量是否为某个构造函数的实例(基于原型链)。 语法:variable instanceof Constructor 返回值:truefalse

特点:

  • 适合判断引用类型(如数组、自定义类)。

  • 无法判断原始类型(如 1 instanceof Numberfalse,除非用 new Number(1) 创建)。

  • 跨框架/窗口(iframe)时会失效,因为不同全局环境下的构造函数不共享原型。

  • 若原型链被修改,结果可能不准确。

3.Object.prototype.toString.call()

作用:返回变量的内部 [[Class]] 类型标识。 语法:Object.prototype.toString.call(variable) 返回值:形如 "[object Type]" 的字符串,如 "[object Array]""[object Null]"

特点:

  • 最全面的类型判断方法,支持所有内置类型(包括 nullundefined)。

  • 可通过 Symbol.toStringTag 自定义类型标签(ES6+),但大部分内置对象不受影响。

  • 对原始类型和引用类型均有效。

适用场景:需要精确判断所有类型(包括ES6+新增类型)。

4. Js的事件循环(Event Loop)是什么

        Js是一种单线程语言(HTML5提供了Web Worker可以开启子线程),同步代码直接交由Js引擎执行,异步代码则交由浏览器或Node.js执行。

        关键概念:

  1. 调用栈(Call Stack)​​:用于跟踪当前正在执行的函数(执行上下文)。当函数被调用时,它被压入栈;当函数返回时,它被弹出栈。
  2. ​任务队列(Task Queue)​​:也称为宏任务队列(Macro Task Queue),包括 setTimeout、setInterval、I/O、事件回调等。
  3. ​微任务队列(Micro Task Queue)​​:包括 Promise 的回调(then/catch/finally)、MutationObserver、queueMicrotask 等。微任务在当前宏任务结束后立即执行,且会清空整个微任务队列,直到队列为空。
  4. ​渲染(Render)步骤​​:在事件循环中,可能会在宏任务和微任务之间进行页面渲染(UI 更新),但这不是事件循环规范的一部分,而是浏览器行为

        事件循环执行过程:

  1. 首先依次执行调用栈中的全部同步代码
  2. 处理微任务:
    • 调用栈清空后会执行全部微任务
    • 微任务执行期间加入的微任务也会顺次执行
  3. 渲染更新(浏览器行为与Js事件循环无关)
  4. 执行一个宏任务
  5. 重复循环

        常见的宏任务与微任务

  • 宏任务
    • setTimeout/setInterval
    • I/O操作
    • 浏览器渲染
    • 事件回调(click等事件)
    • 脚本执行(不同的script标签)
  • 微任务
    • Promise相关方法(Promise.then等)
    • MutationObserver(Dom变化监视)

5.下面代码的输出顺序是什么(事件循环)

async function async1() {console.log('async1 start');await async2(); // 后续为微任务console.log('async1 end');
}async function async2() {console.log('async2');
}console.log('script start');setTimeout(function() {console.log('setTimeout');
}, 0) // 宏任务async1();new Promise(function(resolve) {console.log('promise1');resolve();
}).then(function() { // 微任务console.log('promise2');
});console.log('script end');
// 顺序如下
script start // 第一个同步任务
async1 start // async1方法中的第一个同步任务
async2 // await修饰的方法立即执行,输出async2,async1方法进入微任务队列
promise1 // 执行new Promise,输出promise1,并进入微任务队列
script end // 执行同步代码
async1 end // 执行微任务队列中的第一个任务
promise2 // 执行微任务队列的第二个任务
setTimeout // 执行宏任务

6.Js原生如何获取cookie

document.cookie // 结果是一个Json字符串,可使用JSON.parse()转换

7.Promise.all/allSettled/any/race的区别

Promise.all

  • 核心:等待所有 Promise 全部完成(fulfilled),或遇到第一个失败(rejected)。

  • 结果:

    • 全部成功时:返回一个数组,元素为每个 Promise 的成功值(顺序与输入一致)。

    • 遇到失败时:立即拒绝(reject),返回第一个失败的 Promise 的原因。

  • 适用场景:多个异步操作强依赖,需要所有结果都成功(如:并发请求多个接口)。

  • 示例:

Promise.all([promise1, promise2]) .then(values => console.log(values)) // 所有成功:[value1, value2].catch(error => console.log(error)); // 任一失败(返回第一个错误)

Promise.allSettled

  • 核心:等待所有 Promise 全部完成(无论成功或失败)。

  • 结果:永远成功(resolve),返回一个数组,每个元素为对象:

    • { status: "fulfilled", value: <result> }(成功)

    • { status: "rejected", reason: <error> }(失败)

  • 适用场景:需知道每个操作的最终状态(无论成功失败)(如:批量操作后汇总结果)。

  • 示例:

Promise.allSettled([promise1, promise2]).then(results => { results.forEach(result => { if (result.status === "fulfilled") console.log(result.value); else console.error(result.reason); }); });

Promise.any

  • 核心:等待第一个 成功(fulfilled)的 Promise,或所有都失败。

  • 结果:

    • 任一成功时:返回第一个成功的值。

    • 全部失败时:拒绝(reject)并抛出 AggregateError,包含所有错误信息。

  • 适用场景:获取最快成功的操作(如:多服务器请求,取最快响应)。

  • 示例:

Promise.any([promise1, promise2]).then(value => console.log(value)) // 第一个成功的值.catch(errors => console.log(errors)); // 所有失败(AggregateError 包含错误数组)

Promise.race

  • 核心:等待第一个 完成(无论成功或失败)的 Promise。

  • 结果:返回第一个完成的 Promise 的结果(可能是成功值或失败原因)。

  • 适用场景:竞速场景(如:请求超时控制)。

  • 示例:

// 实现超时控制 
const timeout = new Promise((_, reject) =>setTimeout(() => reject("Timeout"), 1000) 
); 
Promise.race([fetchData(), timeout]).then(data => console.log(data)) // 在超时前完成.catch(error => console.log(error)); // 超时或请求失败

对比表格

方法行为描述成功条件失败条件返回值类型
​​Promise.all​​所有成功或任一失败(短路)全部成功第一个失败数组(成功值)
​​Promise.allSettled​​等待所有完成(无论成败)永不失败-对象数组(包含状态和值/原因)
​​Promise.any​​等待第一个成功或所有失败第一个成功全部失败第一个成功的值
​​Promise.race​​等待第一个完成(无论成败)第一个完成的结果是成功第一个完成的结果是失败第一个完成的结果

关键区别

  • all vs allSettledall 会在遇到失败时立即终止,而 allSettled 始终等待所有结果。

  • any vs raceany 只关心第一个成功(忽略失败),race 关心第一个完成(无论成败)。

  • 错误处理: any 在所有失败时返回 AggregateError;其他方法直接返回单条错误原因。

根据实际需求选择合适的方法:需要全部成功用 all,容忍失败用 allSettled,取最快成功用 any,竞速场景用 race

 8.原型对象、构造函数、实例之间的关系

对象指向关系属性/方法
​​构造函数​​prototype → 原型对象Person.prototype
​​原型对象​​constructor → 构造函数Person.prototype.constructor
​​实例​​__proto__ → 原型对象person.__proto__

9.Proxy和Object.defineProperty的区别 

  • Object.defineProperty: 只能对已存在的属性进行劫持,无法拦截新增的属性和删除的属性(需要通过Vue.setVue.delete实现响应式),由于劫持基于属性级别,对于大规模对象或者数组来说会导致性能下降;(Vue2响应式的实现方法)

  • Proxy: 劫持整个对象,并返回一个代理对象,提高了初始化性能,同时可以拦截新增属性和删除属性。(Vue3响应式的实现方法)

10.使用Proxy代理一个对象,是对这个对象所有层级进行了劫持吗

        不会,proxy只代理外层对象,内层对象需要单独proxy代理

 11.简单数组去重的方法

  • 使用Set只运行存储唯一值的特点
const arr = [1, 2, 2, 3, 3, 4, 5, 5, 6]
const uniqueArr = [...new Set(arr)] 
// 或 const uniqueArr = Array.from(new Set(arr))
  • 使用includes方法检查新数值中是否存在该元素

const arr = [1, 2, 2, 3, 3, 4, 5, 5, 6] 
const uniqueArr = [] 
arr.forEach(val => { if(!uniqueArr.includes(val)) { uniqueArr.push(val) } 
}
  • 使用indexOf方法,检查新数组中是否存在该元素

const arr = [1, 2, 2, 3, 3, 4, 5, 5, 6]
const uniqueArr = [] 
arr.forEach(val => { if(uniqueArr.indexOf(val) === -1) { uniqueArr.push(val) } 
}
  • 使用filter方法过滤第一次出现的元素

const arr = [1, 2, 2, 3, 3, 4, 5, 5, 6]
const uniqueArr = arr.filter((val, index, self) => self.indexOf(val) === index
}
  • 使用reduce方法遍历数组元素

const arr = [1, 2, 2, 3, 3, 4, 5, 5, 6]
const uniqueArr = arr.reduce((res, current) => {if(!res.includes(current) {res.push(current) } return res 
}, [])

12.对象数组如何去重

  • id(或其他键相同算为重复)

    • 使用Map

function uniqueById(arr) { return [ ...new Map( arr.map(item => [item.id, item]) ).values() ]; 
} // 使用示例 
const users = [ { id: 1, name: 'John' },{ id: 2, name: 'Jane' }, { id: 1, name: 'Johnny' }, // 相同id { id: 3, name: 'Alice' } 
]; 
console.log(uniqueById(users)); 
// [ 
// { id: 1, name: 'Johnny' }, 
// { id: 2, name: 'Jane' }, 
// { id: 3, name: 'Alice' } 
// ]
  • 使用filter+缓存 
function uniqueById(arr) { const seen = new Set(); return arr.filter(item => { const duplicate = seen.has(item.id); seen.add(item.id);     return !duplicate; }); 
}
  •  使用reduce 
function uniqueById(arr) { return Object.values(arr.reduce((acc, item) => { acc[item.id] = item; return acc; }, {}) ); 
}
  • 完全相同算重复
    • 使用JSON.stringify(简单对象)
      
function deepUnique(arr) { const seen = new Set(); return arr.filter(item => { const serialized = JSON.stringify(item); return seen.has(serialized) ? false : seen.add(serialized); }); 
} 
// 使用示例 
const items = [ { id: 1, name: 'Apple' }, { id: 2, name: 'Banana' }, { id: 1, name: 'Apple' }, // 相同对象 { id: 1, name: 'apple' } // 不同对象 
]; 
console.log(deepUnique(items)); 
// [ 
// { id: 1, name: 'Apple' }, 
// { id: 2, name: 'Banana' }, 
// { id: 1, name: 'apple' } 
// ]
  •  Lodash的isEqual方法(复杂对象推荐)                 
// 需要安装lodash:npm install lodash 
import _ from 'lodash'; 
function deepUnique(arr) { return arr.reduce((acc, item) => { const isDuplicate = acc.some(accItem => _.isEqual(accItem, item)); return isDuplicate ? acc : [...acc, item]; }, []); 
}

创建于2025.6.2,后续继续更新


http://www.hkcw.cn/article/FIXivdBTrx.shtml

相关文章

花卉目标检测数据集介绍(共 12 类,10490 张图像)

在计算机视觉与智能农业快速发展的背景下&#xff0c;基于深度学习的花卉识别与检测技术正被广泛应用于植物分类、智能园艺、自动监测与生态研究等多个领域。为了推动花卉类目标检测任务的发展&#xff0c;本文介绍一个包含 12 种常见花卉类别的目标检测数据集&#xff0c;总计…

linux学习第18天(fork函数)

pid_t fork(void)&#xff1a;创建一个子进程 成功&#xff1a;父进程返回子进程pid 子进程返回0&#xff08;创建成功&#xff09; 失败&#xff1a;-1 getpid/getppid 举个例子&#xff0c;顺便演示getpid和getppid 子进程只能执行fork之后的代码&#xff0c;并且同时争夺c…

Pycharm的终端无法使用Anaconda命令行问题详细解决教程

很多初学者在Windows系统上安装了Anaconda后&#xff0c;在PyCharm终端中运行Conda命令时&#xff0c;会遇到以下错误&#xff1a; conda : 无法将“conda”项识别为 cmdlet、函数、脚本文件或可运行程序的名称。 请检查名称的拼写&#xff0c;如果包括路径&#xff0c;请确保…

关税政策推进受阻 特朗普政府求助上诉法院

当地时间6月2日,美国特朗普政府请求联邦上诉法院阻止此前哥伦比亚特区联邦地区法院裁定其关税政策“违法”的命令。当地时间5月29日,位于首都华盛顿的哥伦比亚特区联邦地区法院就特朗普政府依据《国际紧急经济权力法》对多国加征的关税措施发布初步禁令。该裁决针对美国两家小…

如何看待乌称摧毁大量俄战略轰炸机 俄方反驳称战果夸大

乌克兰国家安全局6月1日在社交媒体上宣布,对位于西伯利亚地区的俄罗斯轰炸机基地进行了袭击。乌方表示,此次行动目标精准,约34%的俄罗斯战略轰炸机遭到攻击,俄方战略航空兵损失高达70亿美元。行动由乌克兰总统泽连斯基亲自协调,安全局局长马柳克率队实施。泽连斯基高度评价…

刘若钒:很遗憾因伤不能和大家一起并肩战斗了,兄弟们加油 伤病阻挡国家队梦想

北京时间6月2日,中国男足公布了出征印尼客场的25人名单,刘若钒因伤未能入选。刘若钒在社交媒体上表达了遗憾:“重回国家队是对我这半年在海港表现的认可,但很遗憾因伤不能和大家一起并肩战斗了。”他同时表示,为国家队比赛一直是他的梦想,希望队友们加油,为中国队助威。…

俄乌谈判草草结束 细看条件都够狠 双方态度强硬

乌克兰在发动无人机袭击后,与俄罗斯在土耳其进行了第二轮和谈。谈判持续了大约一个小时便匆匆结束。土耳其方面表示,谈判以“非消极”的方式结束。5月16日进行了第一轮谈判,6月2日进行了第二轮谈判。预计第三轮谈判将在两周后进行,期间双方可能还会继续交战。双方态度都十分…

山东10岁失联男孩确认溺水身亡 搜救努力未果

6月2日,滕州市公安局发布警情通报。5月31日22时35分许,孔某某报警称其外孙赵某某(10岁)于当日17时许离家后失联。接警后,公安机关迅速调阅监控、走访群众,并联合当地政府和社会救援力量,采用搜救警犬和无人机等手段持续开展搜寻。6月2日15时许,在邻村一河道内发现赵某某…

PID项目-记事本不显示下划线

在安装安装keil5的插件的时候想要更改路径&#xff0c;用记事本打开的时候下划线都不见了&#xff0c;使用缩放&#xff0c;把100%缩放成90%就又出现了

AIGC工具平台-GPT-SoVITS-v4-TTS音频推理克隆

声音克隆与语音合成的结合&#xff0c;是近年来生成式AI在多模态方向上的重要落地场景之一。随着预训练模型能力的增强&#xff0c;结合语音识别、音素映射与TTS合成的端到端系统成为初学者可以上手实践的全流程方案。 围绕 GPT-SoVITS-v4-TTS 模块&#xff0c;介绍了其在整合…

Cherry Studio 和 Dify 如何接入MCP 服务

这里以魔搭社区的MCP 广场为例,进行介绍 一、Cherry Studio 接入 MCP服务教程 1. 第一步:访问魔搭社区官网 魔搭社区官网 点击上方链接进入魔搭社区官网后,点击MCP广场。 2.第二步:选择要接入的MCP服务(这里以 ‘今天吃什么’ 为例) 点击图中标注的连接,然后点击下…

微服务 Feign相关

1.feign 调用超时控制逻辑 openfign 中配置默认的链接时间是 10S 读取时间是60S 可以在YML文件中修改配置 spring: cloud:nacos:server-addr: 127.0.0.1:8848config:namespace: 51d656b5-cfe8-4cd0-95ad-91811cd88dc6#openfeign 相关配置openfeign:client:config: # …

一键解决Github无法访问或时断时续的问题-Linux环境

创建脚本github_host.sh vi github_host.sh 按i插入文本&#xff0c;将下面一段内容复制进去 #!/bin/bashhosts_path/etc/hosts # 系统 hosts 保存路径 hosts_path_bak/etc/hosts.bak # 系统 hosts 备份路径# 备份 hosts echo "########## 备份 $hosts_path 到 $…

关于海洋数据上云的一些机遇与挑战

知识星球&#xff1a;数据书局。打算通过知识星球将这些年积累的知识、经验分享出来&#xff0c;让各位在数据治理、数据分析的路上少走弯路&#xff0c;另外星球也方便动态更新最近的资料&#xff0c;提供各位一起讨论数据的小圈子 1. 摘要 海洋观测技术的革新以及数据模型…

高效微调方法简述

高效微调简述 一、微调与RAG的区别&#xff1a; 针对成本和性价比选择RAG或微调&#xff0c;那他们适用的范围和区别要提前了解&#xff1b; 形象的描述预训练、微调、提示工程、Agents&#xff1a; 微调流程&#xff1a; 数据保密那就是私域微调&#xff0c;否则就可以线上…

通用优势估计函数(GAE,Generalized Advantage Estimation)详解

强化学习中用于估计优势函数的核心技术&#xff0c;由Schulman等人于2016年提出。核心作用是在强化学习中&#xff0c;用单一可调参数&#xff08;λ&#xff09;平衡偏差与方差&#xff0c;为策略梯度算法&#xff08;如PPO、TRPO、A2C&#xff09;提供稳定、高效的优势函数估…

Java从入门到精通 - 常用API(一)

常用 API 此笔记参考黑马教程&#xff0c;仅学习使用&#xff0c;如有侵权&#xff0c;联系必删 文章目录 常用 API1. 包代码演示 2. String2.1 String 概述代码演示总结 2.2 String 的常用方法代码演示 2.3 String 使用时的注意事项第一点第二点代码演示 总结题目 2.4 String…

n8n部署工作流websecscan-ai-powered-website-security-auditor

人工智能驱动的网站安全审计工作流 此 n8n 工作流程利用DeepSeek V3的模型检测漏洞、配置问题和安全配置错误&#xff0c;提供全面的网站安全分析。该工作流程生成专业的 HTML 安全报告&#xff0c;并直接通过 QQ邮箱 发送。 1.主要特点 双层安全分析&#xff1a;使用专门的…

TC3xx学习笔记-启动过程详解(一)

文章目录 前言Firmware启动过程BMHD Check流程ABM启动Internal Flash启动Bootloader ModeProcessing in case no valid BMHD foundProcessing in case no Boot Mode configured by SSW 总结 前言 之前介绍过UCB BMHD的使用&#xff0c;它在启动过程中起着重要的作用&#xff0…

Docker 镜像制作

目录 镜像制作及原因 快照方式制作镜像 Dockerfile 制作镜像 为什么需要 Dockerfile Dockerfile 指令 常见问题 镜像制作及原因 镜像制作是因为某种需求&#xff0c;官方的镜像无法满足需求&#xff0c;需要我们通过一定手段来自定义镜像来满足要求。 制作镜像往往因为以…